diff --git a/.gitattributes b/.gitattributes index 6313b56c5..d3877a538 100644 --- a/.gitattributes +++ b/.gitattributes @@ -1 +1,2 @@ * text=auto eol=lf +*.svg binary diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100644 index 000000000..490051876 --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1 @@ +github: iliakan diff --git a/.gitignore b/.gitignore index 6f90fd190..1a71fb7c8 100644 --- a/.gitignore +++ b/.gitignore @@ -21,3 +21,4 @@ sftp-config.json Thumbs.db +/svgs \ No newline at end of file diff --git a/1-js/01-getting-started/1-intro/article.md b/1-js/01-getting-started/1-intro/article.md index 727cf4ac3..8af048b43 100644 --- a/1-js/01-getting-started/1-intro/article.md +++ b/1-js/01-getting-started/1-intro/article.md @@ -1,12 +1,18 @@ # En introduksjon til JavaScript -Let's see what's so special about JavaScript, what we can achieve with it, and which other technologies play well with it. +Let's see what's so special about JavaScript, what we can achieve with it, and what other technologies play well with it. La oss se litt nærmere på hva som er så spesielt med JavaScript, hva vi kan oppnå ved å bruke det, og hvilke andre teknologier som passer godt sammen med det. +<<<<<<< HEAD ## Hva er JavaScript? *JavaScript* var egentlig ment for å *"gi hjemmesider mer liv"*. +======= +*JavaScript* was initially created to "make web pages alive". + +The programs in this language are called *scripts*. They can be written right in a web page's HTML and run automatically as the page loads. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Programmer skrevet i JavaScript kalles for *scripts*. De kan skrives rett inn i en nettsides's HTML og kjører automatisk så fort siden blir lastet. @@ -14,8 +20,13 @@ Scripts er utdelt og kjørt som ren tekst. De trenger altså ingen forberedelser Med dette øyemed, er JavaScript veldig annerledes fra et annet språk kalt [Java](https://en.wikipedia.org/wiki/Java_(programming_language)). +<<<<<<< HEAD ```smart header="Why <u>Java</u>Script?" Når JavaScript ble skapt, hadde det i utgangspunktet navnet: "LiveScript". Men Java var veldig populært på den tiden, så det ble bestemt at å presentere et nytt programmeringsspråk som en "lillebror" til Java ville være en god ide. +======= +```smart header="Why is it called <u>Java</u>Script?" +When JavaScript was created, it initially had another name: "LiveScript". But Java was very popular at that time, so it was decided that positioning a new language as a "younger brother" of Java would help. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Men etterhvert som det utviklet seg ble JavaScript et helt uavhengig programmeringsspårk med sitt eget sett med spesifikasjoner kalt [ECMAScript](http://en.wikipedia.org/wiki/ECMAScript), og nå har det ingenting med Java å gjøre i det hele tatt. ``` @@ -27,27 +38,47 @@ Nettleseren har en innebygget motor som noenganger blir kalt for en "JavaScript Different engines have different "codenames". For example: Det finnes flere forskjellige motorer som har skilles fra hverandre med forskjellige "kodenavn". For eksempel: +<<<<<<< HEAD - [V8](https://en.wikipedia.org/wiki/V8_(JavaScript_engine)) -- i Chrome og Opera. - [SpiderMonkey](https://en.wikipedia.org/wiki/SpiderMonkey) -- i Firefox. - ...Det er også andre kodenavn som "Trident" og "Chakra" for forskjellige versjoner av IE, "ChakraCore" for Microsoft Edge, "Nitro" og "SquirrelFish" for Safari, osv. Begrepene ovenfor er gode å huske fordi de ofte brukt i artikler skrevet av utviklere på nettet. Vi vil bruke de også. For eksempel, hvis "en funksjon x er støttet av v8", vil den sannsynlig fungere i Chrome og Opera. +======= +- [V8](https://en.wikipedia.org/wiki/V8_(JavaScript_engine)) -- in Chrome, Opera and Edge. +- [SpiderMonkey](https://en.wikipedia.org/wiki/SpiderMonkey) -- in Firefox. +- ...There are other codenames like "Chakra" for IE, "JavaScriptCore", "Nitro" and "SquirrelFish" for Safari, etc. + +The terms above are good to remember because they are used in developer articles on the internet. We'll use them too. For instance, if "a feature X is supported by V8", then it probably works in Chrome, Opera and Edge. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```smart header="How do engines work?" Motorer er kompliserte. Men det grunnleggende er ganske lett. +<<<<<<< HEAD 1. Motoren (Innebygd hvis det er en nettleser) leser ("parser") scriptet. 2. Så konverterer den ("kompilerer") scriptet til maskin språk 3. Og så kjører maskin koden, veldig raskt. Motoren tar ibruk optimaliseringer til hver eneste steg i prosessen. Passer til og med på det kompilerte scriptet imens det kjører, analyserer data som flyter gjennom det, og legger så til optimaliseringer til maskin koden basert på denne kunnskapen. Når dette gjøres, kjører scripts rimelig fort. +======= +1. The engine (embedded if it's a browser) reads ("parses") the script. +2. Then it converts ("compiles") the script to machine code. +3. And then the machine code runs, pretty fast. + +The engine applies optimizations at each step of the process. It even watches the compiled script as it runs, analyzes the data that flows through it, and further optimizes the machine code based on that knowledge. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ``` ## Hva kan JavaScript gjøre i nettleseren? +<<<<<<< HEAD Moderne JavaScript er et "trygt" programmeringsspråk. Det gir ikke lavniå tilgang til minne eller CPU, fordi det i utgangspunktet var laget for nettlesere som ikke krever det. +======= +Modern JavaScript is a "safe" programming language. It does not provide low-level access to memory or the CPU, because it was initially created for browsers which do not require it. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b JavaScript's evber er sterkt avhengig av miljøet det kjører i. For eksempel, [Node.js](https://wikipedia.org/wiki/Node.js) støtter funksjoner som lar JavaScript lese/skrive vilkårlig, gjennomføre nettverksforespørsler, osv. @@ -63,14 +94,23 @@ For eksempel, i-nettleser JavaScript er i stand til å: ## Hva KAN IKKE JavaScript gjøre i nettleseren? +<<<<<<< HEAD JavaScript's egenskaper i nettleseren er begrensede på vegne av brukeren trygghet. Dette er ment for å hindre nettsider med onde hensikter fra å aksessere informasjon eller ramme brukerens data på noen som helst måte. +======= +JavaScript's abilities in the browser are limited to protect the user's safety. The aim is to prevent an evil webpage from accessing private information or harming the user's data. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Eksempler på slike begrensninger inkluderer: +<<<<<<< HEAD - JavaScript på en nettside kan ikke lese/skrive vilkårlige filer på en harddisk, kopiere eller kjøre programmer. Det har ingen direkte tilgang til funksjoner som er tilknyttet operativsystemet (OS). +======= +- JavaScript on a webpage may not read/write arbitrary files on the hard disk, copy them or execute programs. It has no direct access to OS functions. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Moderne nettlesere tillater det til å behandle filer, men denne adgangen er begrenset og kun gitt hvis brukeren gjennomfører utvalgte hendelser, som å "slippe" en fil over et nettleservindu eller velger det via en `<input>` tag. +<<<<<<< HEAD Det finnes måter å interragere med kamera og mikrofon og andre enheter, men disse krever at brukeren gir eksplisitt tillatelse. Så en JavaScript-akrivert side kan ikke på en lumsk måte aktivere webkameraet, se seg rundt og sende informasjonen om dette til [PST](https://no.wikipedia.org/wiki/Politiets_sikkerhetstjeneste). - Forskjellige paneler/vinduer i en nettleser vet i utgangspunktet ingenting om hverandre. Noenganger gjør de det, for eksempel når et vindu bruker JavaScript til å åpne et annet vindu. Men selv i dette tilfellet, JavaScript fra en side til en kan ikke aksessere et annet vindu hvis det kommer fra forskjellige sider (fra et annet domene, protokoll eller port). @@ -80,6 +120,19 @@ Eksempler på slike begrensninger inkluderer: Denne begrensningen er, igjen satt på plass med brukerens trygghet i tankene. En side fra `http://hvilkensomhelstside.no` som en bruker har åpnet kan overhodet ikke være i stand til å aksessere et annet panel med URL'en `http://gmail.com` og stjele informasjon derfra. - JavaScript kan med letthet kommunisere med en tjener over nettet hvor den nåværende siden kom fra. Men dets egenskap til å motta data fra andre sider/domener er veldig tungvint, men det er mulig, dette krever eksplisitt tillatelse (uttrykt via HTTP headere) fra den eksterne siden. Igjen, dette er sikkerhetsbegrensning. +======= + There are ways to interact with the camera/microphone and other devices, but they require a user's explicit permission. So a JavaScript-enabled page may not sneakily enable a web-camera, observe the surroundings and send the information to the [NSA](https://en.wikipedia.org/wiki/National_Security_Agency). +- Different tabs/windows generally do not know about each other. Sometimes they do, for example when one window uses JavaScript to open the other one. But even in this case, JavaScript from one page may not access the other page if they come from different sites (from a different domain, protocol or port). + + This is called the "Same Origin Policy". To work around that, *both pages* must agree for data exchange and must contain special JavaScript code that handles it. We'll cover that in the tutorial. + + This limitation is, again, for the user's safety. A page from `http://anysite.com` which a user has opened must not be able to access another browser tab with the URL `http://gmail.com`, for example, and steal information from there. +- JavaScript can easily communicate over the net to the server where the current page came from. But its ability to receive data from other sites/domains is crippled. Though possible, it requires explicit agreement (expressed in HTTP headers) from the remote side. Once again, that's a safety limitation. + + + +Such limitations do not exist if JavaScript is used outside of the browser, for example on a server. Modern browsers also allow plugins/extensions which may ask for extended permissions. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  @@ -90,17 +143,27 @@ Slike begrensninger eksisterer ikke hvis JavaScript brukes på utsiden av nettle Dette er de *tre* minst gode ting om JavaScript: ```compare +<<<<<<< HEAD + Full integrasjon med HTML/CSS. + Enkle ting kan gjøres på en enkel måte. + Støttet av alle de største nettleserne og er påslått som standard. +======= ++ Full integration with HTML/CSS. ++ Simple things are done simply. ++ Supported by all major browsers and enabled by default. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ``` JavaScript er den eneste nettleser teknologien som kombinerer disse tre tingene. Dette er det som gjør JavaScript unikt. Dette er grunnen til at JavaScript er det mest utstrakte verktøyet for utvikling av grensesnitt i nettleseren. +<<<<<<< HEAD Imens du har planer om å lære deg en teknologi, er det gunstig å sjekke dens perspektiver. Så la oss fortsette til de moderne trendene som påvirker den, inkluderende nye språk og ny nettleser-funksjonalitet. ## Språk "via" JavaScript +======= +That said, JavaScript can be used to create servers, mobile applications, etc. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b JavaScript's syntax er tilpasset alle sitt behov. Forskjellige folk trenger forskjellige funksjoner. @@ -108,7 +171,11 @@ Dette er ikke overraskende, siden forskjellige prosjekter har forskjellige krav. En haug av nye språk har nylig dukket opp, som er *transpilert* (konvertert) til JavaScript før de kjører i nettleseren. +<<<<<<< HEAD Moderne verktøy gjør transpileringen veldig rask og forutsigbar, faktisk så lar dette utviklere å skrive kode på et annet språk som blir automatisk konvertert "under panseret". +======= +So, recently a plethora of new languages appeared, which are *transpiled* (converted) to JavaScript before they run in the browser. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Dette er eksempler på slike språk: @@ -116,11 +183,28 @@ Dette er eksempler på slike språk: - [TypeScript](http://www.typescriptlang.org/) er konsentrert rundt å legge til "streng datatyping" for å gjøre utvikling og drifting av komplekse systemer lettere. TypeScript er utviklet av Microsoft. - [Dart](https://www.dartlang.org/) er et selvstendig språk som har sin egen motor som kjører ukjente miljøer for nettleseren (som mobilapplikasjoner). Det var i utgangspunktet lagt frem av Google som en erstatter for JavaScript, men foreløpig krever nettlesere at Dart transpileres til JavaScript akkurat som de andre nevnt ovenfor. +<<<<<<< HEAD Det finnes flere selvfølgelig. Men selv om vi bruker noen av disse språkene, er det lurt av oss å kjenne JavaScript for å vite hva vi egentlig driver med. +======= +- [CoffeeScript](https://coffeescript.org/) is "syntactic sugar" for JavaScript. It introduces shorter syntax, allowing us to write clearer and more precise code. Usually, Ruby devs like it. +- [TypeScript](https://www.typescriptlang.org/) is concentrated on adding "strict data typing" to simplify the development and support of complex systems. It is developed by Microsoft. +- [Flow](https://flow.org/) also adds data typing, but in a different way. Developed by Facebook. +- [Dart](https://www.dartlang.org/) is a standalone language that has its own engine that runs in non-browser environments (like mobile apps), but also can be transpiled to JavaScript. Developed by Google. +- [Brython](https://brython.info/) is a Python transpiler to JavaScript that enables the writing of applications in pure Python without JavaScript. +- [Kotlin](https://kotlinlang.org/docs/reference/js-overview.html) is a modern, concise and safe programming language that can target the browser or Node. + +There are more. Of course, even if we use one of these transpiled languages, we should also know JavaScript to really understand what we're doing. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Summary ## Oppsummering +<<<<<<< HEAD - JavaScript var i utgangspunktet skapt som et kun-nettleser språk, men finnes også i mange andre miljøer i tillegg. - Idag, har JavaScript en unik position som det mest brukt nettleser språket med full integrasjon mot HTML/CSS. - Det finnes mange språk som blir "transpilert" til JavaScript og som gir utvalgte funksjoner. Det er anbefalt å ta en kikk på de, ihvertfall ganske kort, etter at du mestrer JavaScript. +======= +- JavaScript was initially created as a browser-only language, but it is now used in many other environments as well. +- Today, JavaScript has a unique position as the most widely-adopted browser language, fully integrated with HTML/CSS. +- There are many languages that get "transpiled" to JavaScript and provide certain features. It is recommended to take a look at them, at least briefly, after mastering JavaScript. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/01-getting-started/2-manuals-specifications/article.md b/1-js/01-getting-started/2-manuals-specifications/article.md new file mode 100644 index 000000000..3fa243336 --- /dev/null +++ b/1-js/01-getting-started/2-manuals-specifications/article.md @@ -0,0 +1,37 @@ + +# Manuals and specifications + +This book is a *tutorial*. It aims to help you gradually learn the language. But once you're familiar with the basics, you'll need other resources. + +## Specification + +[The ECMA-262 specification](https://www.ecma-international.org/publications/standards/Ecma-262.htm) contains the most in-depth, detailed and formalized information about JavaScript. It defines the language. + +But being that formalized, it's difficult to understand at first. So if you need the most trustworthy source of information about the language details, the specification is the right place. But it's not for everyday use. + +A new specification version is released every year. Between these releases, the latest specification draft is at <https://tc39.es/ecma262/>. + +To read about new bleeding-edge features, including those that are "almost standard" (so-called "stage 3"), see proposals at <https://github.com/tc39/proposals>. + +Also, if you're developing for the browser, then there are other specifications covered in the [second part](info:browser-environment) of the tutorial. + +## Manuals + +- **MDN (Mozilla) JavaScript Reference** is the main manual with examples and other information. It's great to get in-depth information about individual language functions, methods etc. + + You can find it at <https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference>. + +Although, it's often best to use an internet search instead. Just use "MDN [term]" in the query, e.g. <https://google.com/search?q=MDN+parseInt> to search for the `parseInt` function. + +## Compatibility tables + +JavaScript is a developing language, new features get added regularly. + +To see their support among browser-based and other engines, see: + +- <https://caniuse.com> - per-feature tables of support, e.g. to see which engines support modern cryptography functions: <https://caniuse.com/#feat=cryptography>. +- <https://kangax.github.io/compat-table> - a table with language features and engines that support those or don't support. + +All these resources are useful in real-life development, as they contain valuable information about language details, their support, etc. + +Please remember them (or this page) for the cases when you need in-depth information about a particular feature. diff --git a/1-js/01-getting-started/2-code-editors/article.md b/1-js/01-getting-started/3-code-editors/article.md similarity index 60% rename from 1-js/01-getting-started/2-code-editors/article.md rename to 1-js/01-getting-started/3-code-editors/article.md index d36561bc6..ca6194741 100644 --- a/1-js/01-getting-started/2-code-editors/article.md +++ b/1-js/01-getting-started/3-code-editors/article.md @@ -12,14 +12,12 @@ An IDE loads the project (which can be many files), allows navigation between fi If you haven't selected an IDE yet, consider the following options: -- [WebStorm](http://www.jetbrains.com/webstorm/) for frontend development. The same company offers other editors for other languages (paid). -- [Netbeans](http://netbeans.org/) (free). +- [Visual Studio Code](https://code.visualstudio.com/) (cross-platform, free). +- [WebStorm](https://www.jetbrains.com/webstorm/) (cross-platform, paid). -All of these IDEs are cross-platform. +For Windows, there's also "Visual Studio", not to be confused with "Visual Studio Code". "Visual Studio" is a paid and mighty Windows-only editor, well-suited for the .NET platform. It's also good at JavaScript. There's also a free version [Visual Studio Community](https://www.visualstudio.com/vs/community/). -For Windows, there's also "Visual Studio", not to be confused with "Visual Studio Code." "Visual Studio" is a paid and mighty Windows-only editor, well-suited for the .NET platform. A free version of it is called [Visual Studio Community](https://www.visualstudio.com/vs/community/). - -Many IDEs are paid but have a trial period. Their cost is usually negligible compared to a qualified developer's salary, so just choose the best one for you. +Many IDEs are paid, but have a trial period. Their cost is usually negligible compared to a qualified developer's salary, so just choose the best one for you. ## Lightweight editors @@ -31,22 +29,11 @@ The main difference between a "lightweight editor" and an "IDE" is that an IDE w In practice, lightweight editors may have a lot of plugins including directory-level syntax analyzers and autocompleters, so there's no strict border between a lightweight editor and an IDE. -The following options deserve your attention: +There are many options, for instance: -- [Visual Studio Code](https://code.visualstudio.com/) (cross-platform, free) also has many IDE-like features. -- [Atom](https://atom.io/) (cross-platform, free). -- [Sublime Text](http://www.sublimetext.com) (cross-platform, shareware). +- [Sublime Text](https://www.sublimetext.com/) (cross-platform, shareware). - [Notepad++](https://notepad-plus-plus.org/) (Windows, free). -- [Vim](http://www.vim.org/) and [Emacs](https://www.gnu.org/software/emacs/) are also cool if you know how to use them. - -## My favorites - -The personal preference of the author is to have both an IDE for projects and a lightweight editor for quick and easy file editing. - -I'm using: - -- As an IDE for JS -- [WebStorm](http://www.jetbrains.com/webstorm/) (I switch to one of the other JetBrains offerings when using other languages) -- As a lightweight editor -- [Sublime Text](http://www.sublimetext.com) or [Atom](https://atom.io/). +- [Vim](https://www.vim.org/) and [Emacs](https://www.gnu.org/software/emacs/) are also cool if you know how to use them. ## Let's not argue @@ -55,3 +42,8 @@ The editors in the lists above are those that either I or my friends whom I cons There are other great editors in our big world. Please choose the one you like the most. The choice of an editor, like any other tool, is individual and depends on your projects, habits, and personal preferences. + +The author's personal opinion: + +- I'd use [Visual Studio Code](https://code.visualstudio.com/) if I develop mostly frontend. +- Otherwise, if it's mostly another language/platform and partially frontend, then consider other editors, such as XCode (Mac), Visual Studio (Windows) or Jetbrains family (Webstorm, PHPStorm, RubyMine etc, depending on the language). diff --git a/1-js/01-getting-started/3-devtools/chrome.png b/1-js/01-getting-started/3-devtools/chrome.png deleted file mode 100644 index 4cb3ea2f4..000000000 Binary files a/1-js/01-getting-started/3-devtools/chrome.png and /dev/null differ diff --git a/1-js/01-getting-started/3-devtools/chrome@2x.png b/1-js/01-getting-started/3-devtools/chrome@2x.png deleted file mode 100644 index b87404a8f..000000000 Binary files a/1-js/01-getting-started/3-devtools/chrome@2x.png and /dev/null differ diff --git a/1-js/01-getting-started/3-devtools/safari.png b/1-js/01-getting-started/3-devtools/safari.png deleted file mode 100644 index 37598a261..000000000 Binary files a/1-js/01-getting-started/3-devtools/safari.png and /dev/null differ diff --git a/1-js/01-getting-started/3-devtools/safari@2x.png b/1-js/01-getting-started/3-devtools/safari@2x.png deleted file mode 100644 index c59cebef2..000000000 Binary files a/1-js/01-getting-started/3-devtools/safari@2x.png and /dev/null differ diff --git a/1-js/01-getting-started/3-devtools/article.md b/1-js/01-getting-started/4-devtools/article.md similarity index 91% rename from 1-js/01-getting-started/3-devtools/article.md rename to 1-js/01-getting-started/4-devtools/article.md index ae5b3845d..bbe8af920 100644 --- a/1-js/01-getting-started/3-devtools/article.md +++ b/1-js/01-getting-started/4-devtools/article.md @@ -22,17 +22,22 @@ The developer tools will open on the Console tab by default. It looks somewhat like this: - + The exact look of developer tools depends on your version of Chrome. It changes from time to time but should be similar. - Here we can see the red-colored error message. In this case, the script contains an unknown "lalala" command. - On the right, there is a clickable link to the source `bug.html:12` with the line number where the error has occurred. -Below the error message, there is a blue `>` symbol. It marks a "command line" where we can type JavaScript commands. Press `key:Enter` to run them (`key:Shift+Enter` to input multi-line commands). +Below the error message, there is a blue `>` symbol. It marks a "command line" where we can type JavaScript commands. Press `key:Enter` to run them. Now we can see errors, and that's enough for a start. We'll come back to developer tools later and cover debugging more in-depth in the chapter <info:debugging-chrome>. +```smart header="Multi-line input" +Usually, when we put a line of code into the console, and then press `key:Enter`, it executes. + +To insert multiple lines, press `key:Shift+Enter`. This way one can enter long fragments of JavaScript code. +``` ## Firefox, Edge, and others @@ -44,18 +49,12 @@ The look & feel of them is quite similar. Once you know how to use one of these Safari (Mac browser, not supported by Windows/Linux) is a little bit special here. We need to enable the "Develop menu" first. -Open Preferences and go to the "Advanced" pane. There's a checkbox at the bottom: +Open Settings and go to the "Advanced" pane. There's a checkbox at the bottom:  Now `key:Cmd+Opt+C` can toggle the console. Also, note that the new top menu item named "Develop" has appeared. It has many commands and options. -## Multi-line input - -Usually, when we put a line of code into the console, and then press `key:Enter`, it executes. - -To insert multiple lines, press `key:Shift+Enter`. - ## Summary - Developer tools allow us to see errors, run commands, examine variables, and much more. diff --git a/1-js/01-getting-started/3-devtools/bug.html b/1-js/01-getting-started/4-devtools/bug.html similarity index 100% rename from 1-js/01-getting-started/3-devtools/bug.html rename to 1-js/01-getting-started/4-devtools/bug.html diff --git a/1-js/01-getting-started/4-devtools/chrome.webp b/1-js/01-getting-started/4-devtools/chrome.webp new file mode 100644 index 000000000..bdf067079 Binary files /dev/null and b/1-js/01-getting-started/4-devtools/chrome.webp differ diff --git a/1-js/01-getting-started/4-devtools/chrome@2.webp b/1-js/01-getting-started/4-devtools/chrome@2.webp new file mode 100644 index 000000000..2aeca5898 Binary files /dev/null and b/1-js/01-getting-started/4-devtools/chrome@2.webp differ diff --git a/1-js/01-getting-started/4-devtools/safari.png b/1-js/01-getting-started/4-devtools/safari.png new file mode 100644 index 000000000..4538827eb Binary files /dev/null and b/1-js/01-getting-started/4-devtools/safari.png differ diff --git a/1-js/01-getting-started/4-devtools/safari@2x.png b/1-js/01-getting-started/4-devtools/safari@2x.png new file mode 100644 index 000000000..1561b2bd9 Binary files /dev/null and b/1-js/01-getting-started/4-devtools/safari@2x.png differ diff --git a/1-js/02-first-steps/01-hello-world/1-hello-alert/index.html b/1-js/02-first-steps/01-hello-world/1-hello-alert/index.html new file mode 100644 index 000000000..ff1d871b0 --- /dev/null +++ b/1-js/02-first-steps/01-hello-world/1-hello-alert/index.html @@ -0,0 +1,12 @@ +<!DOCTYPE html> +<html> + +<body> + + <script> + alert( "I'm JavaScript!" ); + </script> + +</body> + +</html> diff --git a/1-js/02-first-steps/01-hello-world/1-hello-alert/solution.md b/1-js/02-first-steps/01-hello-world/1-hello-alert/solution.md index e69de29bb..81552913b 100644 --- a/1-js/02-first-steps/01-hello-world/1-hello-alert/solution.md +++ b/1-js/02-first-steps/01-hello-world/1-hello-alert/solution.md @@ -0,0 +1,2 @@ + +[html src="index.html"] diff --git a/1-js/02-first-steps/01-hello-world/article.md b/1-js/02-first-steps/01-hello-world/article.md index d96ffd76f..35f82bf5d 100644 --- a/1-js/02-first-steps/01-hello-world/article.md +++ b/1-js/02-first-steps/01-hello-world/article.md @@ -1,6 +1,6 @@ # Hello, world! -The tutorial that you're reading is about core JavaScript, which is platform-independent. Later on, you'll learn about Node.js and other platforms that use it. +This part of the tutorial is about core JavaScript, the language itself. But we need a working environment to run our scripts and, since this book is online, the browser is a good choice. We'll keep the amount of browser-specific commands (like `alert`) to a minimum so that you don't spend time on them if you plan to concentrate on another environment (like Node.js). We'll focus on JavaScript in the browser in the [next part](/ui) of the tutorial. @@ -9,7 +9,7 @@ So first, let's see how we attach a script to a webpage. For server-side environ ## The "script" tag -JavaScript programs can be inserted into any part of an HTML document with the help of the `<script>` tag. +JavaScript programs can be inserted almost anywhere into an HTML document using the `<script>` tag. For instance: @@ -46,7 +46,7 @@ The `<script>` tag contains JavaScript code which is automatically executed when The `<script>` tag has a few attributes that are rarely used nowadays but can still be found in old code: The `type` attribute: <code><script <u>type</u>=...></code> -: The old HTML standard, HTML4, required a script to have a `type`. Usually it was `type="text/javascript"`. It's not required anymore. Also, the modern HTML standard, HTML5, totally changed the meaning of this attribute. Now, it can be used for JavaScript modules. But that's an advanced topic; we'll talk about modules in another part of the tutorial. +: The old HTML standard, HTML4, required a script to have a `type`. Usually it was `type="text/javascript"`. It's not required anymore. Also, the modern HTML standard totally changed the meaning of this attribute. Now, it can be used for JavaScript modules. But that's an advanced topic, we'll talk about modules in another part of the tutorial. The `language` attribute: <code><script <u>language</u>=...></code> : This attribute was meant to show the language of the script. This attribute no longer makes sense because JavaScript is the default language. There is no need to use it. @@ -60,7 +60,7 @@ Comments before and after scripts. //--></script> ``` - This trick isn't used in modern JavaScript. These comments hid JavaScript code from old browsers that didn't know how to process the `<script>` tag. Since browsers released in the last 15 years don't have this issue, this kind of comment can help you identify really old code. + This trick isn't used in modern JavaScript. These comments hide JavaScript code from old browsers that didn't know how to process the `<script>` tag. Since browsers released in the last 15 years don't have this issue, this kind of comment can help you identify really old code. ## External scripts @@ -73,14 +73,12 @@ Script files are attached to HTML with the `src` attribute: <script src="/path/to/script.js"></script> ``` -Here, `/path/to/script.js` is an absolute path to the script file (from the site root). - -You can also provide a relative path from the current page. For instance, `src="script.js"` would mean a file `"script.js"` in the current folder. +Here, `/path/to/script.js` is an absolute path to the script from the site root. One can also provide a relative path from the current page. For instance, `src="script.js"`, just like `src="./script.js"`, would mean a file `"script.js"` in the current folder. We can give a full URL as well. For instance: ```html -<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/3.2.0/lodash.js"></script> +<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.11/lodash.js"></script> ``` To attach several scripts, use multiple tags: diff --git a/1-js/02-first-steps/02-structure/article.md b/1-js/02-first-steps/02-structure/article.md index b18aab19e..e81fd343d 100644 --- a/1-js/02-first-steps/02-structure/article.md +++ b/1-js/02-first-steps/02-structure/article.md @@ -46,7 +46,7 @@ alert(3 + + 2); ``` -The code outputs `6` because JavaScript does not insert semicolons here. It is intuitively obvious that if the line ends with a plus `"+"`, then it is an "incomplete expression", so the semicolon is not required. And in this case that works as intended. +The code outputs `6` because JavaScript does not insert semicolons here. It is intuitively obvious that if the line ends with a plus `"+"`, then it is an "incomplete expression", so a semicolon there would be incorrect. And in this case, that works as intended. **But there are situations where JavaScript "fails" to assume a semicolon where it is really needed.** @@ -56,45 +56,41 @@ Errors which occur in such cases are quite hard to find and fix. If you're curious to see a concrete example of such an error, check this code out: ```js run -[1, 2].forEach(alert) +alert("Hello"); + +[1, 2].forEach(alert); ``` -No need to think about the meaning of the brackets `[]` and `forEach` yet. We'll study them later. For now, just remember the result of the code: it shows `1` then `2`. +No need to think about the meaning of the brackets `[]` and `forEach` yet. We'll study them later. For now, just remember the result of running the code: it shows `Hello`, then `1`, then `2`. -Now, let's add an `alert` before the code and *not* finish it with a semicolon: +Now let's remove the semicolon after the `alert`: ```js run no-beautify -alert("There will be an error") +alert("Hello") -[1, 2].forEach(alert) +[1, 2].forEach(alert); ``` -Now if we run the code, only the first `alert` is shown and then we have an error! - -But everything is fine again if we add a semicolon after `alert`: -```js run -alert("All fine now"); +The difference compared to the code above is only one character: the semicolon at the end of the first line is gone. -[1, 2].forEach(alert) -``` +If we run this code, only the first `Hello` shows (and there's an error, you may need to open the console to see it). There are no numbers any more. -Now we have the "All fine now" message followed by `1` and `2`. +That's because JavaScript does not assume a semicolon before square brackets `[...]`. So, the code in the last example is treated as a single statement. - -The error in the no-semicolon variant occurs because JavaScript does not assume a semicolon before square brackets `[...]`. - -So, because the semicolon is not auto-inserted, the code in the first example is treated as a single statement. Here's how the engine sees it: +Here's how the engine sees it: ```js run no-beautify -alert("There will be an error")[1, 2].forEach(alert) +alert("Hello")[1, 2].forEach(alert); ``` -But it should be two separate statements, not one. Such a merging in this case is just wrong, hence the error. This can happen in other situations. +Looks weird, right? Such merging in this case is just wrong. We need to put a semicolon after `alert` for the code to work correctly. + +This can happen in other situations also. ```` We recommend putting semicolons between statements even if they are separated by newlines. This rule is widely adopted by the community. Let's note once again -- *it is possible* to leave out semicolons most of the time. But it's safer -- especially for a beginner -- to use them. -## Comments +## Comments [#code-comments] As time goes on, programs become more and more complex. It becomes necessary to add *comments* which describe what the code does and why. @@ -136,7 +132,7 @@ alert('World'); ``` ```smart header="Use hotkeys!" -In most editors, a line of code can be commented out by pressing the `key:Ctrl+/` hotkey for a single-line comment and something like `key:Ctrl+Shift+/` -- for multiline comments (select a piece of code and press the hotkey). For Mac, try `key:Cmd` instead of `key:Ctrl`. +In most editors, a line of code can be commented out by pressing the `key:Ctrl+/` hotkey for a single-line comment and something like `key:Ctrl+Shift+/` -- for multiline comments (select a piece of code and press the hotkey). For Mac, try `key:Cmd` instead of `key:Ctrl` and `key:Option` instead of `key:Shift`. ``` ````warn header="Nested comments are not supported!" diff --git a/1-js/02-first-steps/03-strict-mode/article.md b/1-js/02-first-steps/03-strict-mode/article.md index 0aab06893..9586733cc 100644 --- a/1-js/02-first-steps/03-strict-mode/article.md +++ b/1-js/02-first-steps/03-strict-mode/article.md @@ -4,7 +4,7 @@ For a long time, JavaScript evolved without compatibility issues. New features w That had the benefit of never breaking existing code. But the downside was that any mistake or an imperfect decision made by JavaScript's creators got stuck in the language forever. -This was the case until 2009 when ECMAScript 5 (ES5) appeared. It added new features to the language and modified some of the existing ones. To keep the old code working, most modifications are off by default. You need to explicitly enable them with a special directive: `"use strict"`. +This was the case until 2009 when ECMAScript 5 (ES5) appeared. It added new features to the language and modified some of the existing ones. To keep the old code working, most such modifications are off by default. You need to explicitly enable them with a special directive: `"use strict"`. ## "use strict" @@ -19,10 +19,7 @@ For example: ... ``` -We will learn functions (a way to group commands) soon. - -Looking ahead, let's just note that `"use strict"` can be put at the start of most kinds of functions instead of the whole script. Doing that enables strict mode in that function only. But usually, people use it for the whole script. - +Quite soon we're going to learn functions (a way to group commands), so let's note in advance that `"use strict"` can be put at the beginning of a function. Doing that enables strict mode in that function only. But usually people use it for the whole script. ````warn header="Ensure that \"use strict\" is at the top" Please make sure that `"use strict"` is at the top of your scripts, otherwise strict mode may not be enabled. @@ -44,36 +41,49 @@ Only comments may appear above `"use strict"`. ```warn header="There's no way to cancel `use strict`" There is no directive like `"no use strict"` that reverts the engine to old behavior. -Once we enter strict mode, there's no return. +Once we enter strict mode, there's no going back. ``` ## Browser console -For the future, when you use a browser console to test features, please note that it doesn't `use strict` by default. +When you use a [developer console](info:devtools) to run code, please note that it doesn't `use strict` by default. Sometimes, when `use strict` makes a difference, you'll get incorrect results. -Even if we press `key:Shift+Enter` to input multiple lines, and put `use strict` on top, it doesn't work. That's because of how the console executes the code internally. +So, how to actually `use strict` in the console? + +First, you can try to press `key:Shift+Enter` to input multiple lines, and put `use strict` on top, like this: + +```js +'use strict'; <Shift+Enter for a newline> +// ...your code +<Enter to run> +``` + +It works in most browsers, namely Firefox and Chrome. -The reliable way to ensure `use strict` would be to input the code into console like this: +If it doesn't, e.g. in an old browser, there's an ugly, but reliable way to ensure `use strict`. Put it inside this kind of wrapper: ```js (function() { 'use strict'; - // ...your code... + // ...your code here... })() ``` -## Always "use strict" +## Should we "use strict"? + +The question may sound obvious, but it's not so. + +One could recommend to start scripts with `"use strict"`... But you know what's cool? + +Modern JavaScript supports "classes" and "modules" - advanced language structures (we'll surely get to them), that enable `use strict` automatically. So we don't need to add the `"use strict"` directive, if we use them. -We have yet to cover the differences between strict mode and the "default" mode. +**So, for now `"use strict";` is a welcome guest at the top of your scripts. Later, when your code is all in classes and modules, you may omit it.** -In the next chapters, as we learn language features, we'll note the differences between the strict and default modes. Luckily, there aren't many and they actually make our lives better. +As of now, we've got to know about `use strict` in general. -For now, it's enough to know about it in general: +In the next chapters, as we learn language features, we'll see the differences between the strict and old modes. Luckily, there aren't many and they actually make our lives better. -1. The `"use strict"` directive switches the engine to the "modern" mode, changing the behavior of some built-in features. We'll see the details later in the tutorial. -2. Strict mode is enabled by placing `"use strict"` at the top of a script or function. Several language features, like "classes" and "modules", enable strict mode automatically. -3. Strict mode is supported by all modern browsers. -4. We recommended always starting scripts with `"use strict"`. All examples in this tutorial assume strict mode unless (very rarely) specified otherwise. +All examples in this tutorial assume strict mode unless (very rarely) specified otherwise. diff --git a/1-js/02-first-steps/04-variables/2-declare-variables/solution.md b/1-js/02-first-steps/04-variables/2-declare-variables/solution.md index 9ffc3efca..392f4e26f 100644 --- a/1-js/02-first-steps/04-variables/2-declare-variables/solution.md +++ b/1-js/02-first-steps/04-variables/2-declare-variables/solution.md @@ -1,4 +1,4 @@ -First, the variable for the name of our planet. +## The variable for our planet That's simple: @@ -6,9 +6,9 @@ That's simple: let ourPlanetName = "Earth"; ``` -Note, we could use a shorter name `planet`, but it might be not obvious what planet it refers to. It's nice to be more verbose. At least until the variable isNotTooLong. +Note, we could use a shorter name `planet`, but it might not be obvious what planet it refers to. It's nice to be more verbose. At least until the variable isNotTooLong. -Second, the name of the current visitor: +## The name of the current visitor ```js let currentUserName = "John"; diff --git a/1-js/02-first-steps/04-variables/3-uppercast-constant/solution.md b/1-js/02-first-steps/04-variables/3-uppercast-constant/solution.md index f3a96c692..acd643fde 100644 --- a/1-js/02-first-steps/04-variables/3-uppercast-constant/solution.md +++ b/1-js/02-first-steps/04-variables/3-uppercast-constant/solution.md @@ -2,4 +2,4 @@ We generally use upper case for constants that are "hard-coded". Or, in other wo In this code, `birthday` is exactly like that. So we could use the upper case for it. -In contrast, `age` is evaluated in run-time. Today we have one age, a year after we'll have another one. It is constant in a sense that it does not change through the code execution. But it is a bit "less of a constant" than `birthday`, it is calculated, so we should keep the lower case for it. \ No newline at end of file +In contrast, `age` is evaluated in run-time. Today we have one age, a year after we'll have another one. It is constant in a sense that it does not change through the code execution. But it is a bit "less of a constant" than `birthday`: it is calculated, so we should keep the lower case for it. diff --git a/1-js/02-first-steps/04-variables/3-uppercast-constant/task.md b/1-js/02-first-steps/04-variables/3-uppercast-constant/task.md index 5fd18f90a..f3c208a74 100644 --- a/1-js/02-first-steps/04-variables/3-uppercast-constant/task.md +++ b/1-js/02-first-steps/04-variables/3-uppercast-constant/task.md @@ -12,13 +12,14 @@ const birthday = '18.04.1982'; const age = someCode(birthday); ``` -Here we have a constant `birthday` date and the `age` is calculated from `birthday` with the help of some code (it is not provided for shortness, and because details don't matter here). +Here we have a constant `birthday` for the date, and also the `age` constant. + +The `age` is calculated from `birthday` using `someCode()`, which means a function call that we didn't explain yet (we will soon!), but the details don't matter here, the point is that `age` is calculated somehow based on the `birthday`. Would it be right to use upper case for `birthday`? For `age`? Or even for both? ```js -const BIRTHDAY = '18.04.1982'; // make uppercase? +const BIRTHDAY = '18.04.1982'; // make birthday uppercase? -const AGE = someCode(BIRTHDAY); // make uppercase? +const AGE = someCode(BIRTHDAY); // make age uppercase? ``` - diff --git a/1-js/02-first-steps/04-variables/article.md b/1-js/02-first-steps/04-variables/article.md index ab6c47281..e91d0ea7e 100644 --- a/1-js/02-first-steps/04-variables/article.md +++ b/1-js/02-first-steps/04-variables/article.md @@ -12,7 +12,7 @@ A [variable](https://en.wikipedia.org/wiki/Variable_(computer_science)) is a "na To create a variable in JavaScript, use the `let` keyword. -The statement below creates (in other words: *declares* or *defines*) a variable with the name "message": +The statement below creates (in other words: *declares*) a variable with the name "message": ```js let message; @@ -24,7 +24,7 @@ Now, we can put some data into it by using the assignment operator `=`: let message; *!* -message = 'Hello'; // store the string +message = 'Hello'; // store the string 'Hello' in the variable named message */!* ``` @@ -64,6 +64,7 @@ let message = 'Hello'; ``` Some people also define multiple variables in this multiline style: + ```js no-beautify let user = 'John', age = 25, @@ -80,7 +81,6 @@ let user = 'John' Technically, all these variants do the same thing. So, it's a matter of personal taste and aesthetics. - ````smart header="`var` instead of `let`" In older scripts, you may also find another keyword: `var` instead of `let`: @@ -88,22 +88,23 @@ In older scripts, you may also find another keyword: `var` instead of `let`: *!*var*/!* message = 'Hello'; ``` -The `var` keyword is *almost* the same as `let`. It also declares a variable, but in a slightly different, "old-school" way. +The `var` keyword is *almost* the same as `let`. It also declares a variable but in a slightly different, "old-school" way. -There are subtle differences between `let` and `var`, but they do not matter for us yet. We'll cover them in detail in the chapter <info:var>. +There are subtle differences between `let` and `var`, but they do not matter to us yet. We'll cover them in detail in the chapter <info:var>. ```` ## A real-life analogy We can easily grasp the concept of a "variable" if we imagine it as a "box" for data, with a uniquely-named sticker on it. -For instance, the variable `message` can be imagined as a box labeled `"message"` with the value `"Hello!"` in it: +For instance, the variable `message` can be imagined as a box labelled `"message"` with the value `"Hello!"` in it:  We can put any value in the box. We can also change it as many times as we want: + ```js run let message; @@ -135,12 +136,26 @@ alert(hello); // Hello world! alert(message); // Hello world! ``` +````warn header="Declaring twice triggers an error" +A variable should be declared only once. + +A repeated declaration of the same variable is an error: + +```js run +let message = "This"; + +// repeated 'let' leads to an error +let message = "That"; // SyntaxError: 'message' has already been declared +``` +So, we should declare a variable once and then refer to it without `let`. +```` + ```smart header="Functional languages" -It's interesting to note that [functional](https://en.wikipedia.org/wiki/Functional_programming) programming languages, like [Scala](http://www.scala-lang.org/) or [Erlang](http://www.erlang.org/), forbid changing variable values. +It's interesting to note that there exist so-called [pure functional](https://en.wikipedia.org/wiki/Purely_functional_programming) programming languages, such as [Haskell](https://en.wikipedia.org/wiki/Haskell), that forbid changing variable values. In such languages, once the value is stored "in the box", it's there forever. If we need to store something else, the language forces us to create a new box (declare a new variable). We can't reuse the old one. -Though it may seem a little odd at first sight, these languages are quite capable of serious development. More than that, there are areas like parallel computations where this limitation confers certain benefits. Studying such a language (even if you're not planning to use it soon) is recommended to broaden the mind. +Though it may seem a little odd at first sight, these languages are quite capable of serious development. More than that, there are areas like parallel computations where this limitation confers certain benefits. ``` ## Variable naming [#variable-naming] @@ -179,18 +194,18 @@ let my-name; // hyphens '-' aren't allowed in the name ``` ```smart header="Case matters" -Variables named `apple` and `AppLE` are two different variables. +Variables named `apple` and `APPLE` are two different variables. ``` -````smart header="Non-English letters are allowed, but not recommended" -It is possible to use any language, including cyrillic letters or even hieroglyphs, like this: +````smart header="Non-Latin letters are allowed, but not recommended" +It is possible to use any language, including Cyrillic letters, Chinese logograms and so on, like this: ```js let имя = '...'; let 我 = '...'; ``` -Technically, there is no error here, such names are allowed, but there is an international tradition to use English in variable names. Even if we're writing a small script, it may have a long life ahead. People from other countries may need to read it some time. +Technically, there is no error here. Such names are allowed, but there is an international convention to use English in variable names. Even if we're writing a small script, it may have a long life ahead. People from other countries may need to read it sometime. ```` ````warn header="Reserved names" @@ -237,7 +252,7 @@ To declare a constant (unchanging) variable, use `const` instead of `let`: const myBirthday = '18.04.1982'; ``` -Variables declared using `const` are called "constants". They cannot be changed. An attempt to do so would cause an error: +Variables declared using `const` are called "constants". They cannot be reassigned. An attempt to do so would cause an error: ```js run const myBirthday = '18.04.1982'; @@ -245,16 +260,15 @@ const myBirthday = '18.04.1982'; myBirthday = '01.01.2001'; // error, can't reassign the constant! ``` -When a programmer is sure that a variable will never change, they can declare it with `const` to guarantee and clearly communicate that fact to everyone. - +When a programmer is sure that a variable will never change, they can declare it with `const` to guarantee and communicate that fact to everyone. ### Uppercase constants -There is a widespread practice to use constants as aliases for difficult-to-remember values that are known prior to execution. +There is a widespread practice to use constants as aliases for difficult-to-remember values that are known before execution. Such constants are named using capital letters and underscores. -Like this: +For instance, let's make constants for colors in so-called "web" (hexadecimal) format: ```js run const COLOR_RED = "#F00"; @@ -275,35 +289,36 @@ Benefits: When should we use capitals for a constant and when should we name it normally? Let's make that clear. -Being a "constant" just means that a variable's value never changes. But there are constants that are known prior to execution (like a hexadecimal value for red) and there are constants that are *calculated* in run-time, during the execution, but do not change after their initial assignment. +Being a "constant" just means that a variable's value never changes. But some constants are known before execution (like a hexadecimal value for red) and some constants are *calculated* in run-time, during the execution, but do not change after their initial assignment. For instance: + ```js const pageLoadTime = /* time taken by a webpage to load */; ``` -The value of `pageLoadTime` is not known prior to the page load, so it's named normally. But it's still a constant because it doesn't change after assignment. +The value of `pageLoadTime` is not known before the page load, so it's named normally. But it's still a constant because it doesn't change after the assignment. -In other words, capital-named constants are only used as aliases for "hard-coded" values. +In other words, capital-named constants are only used as aliases for "hard-coded" values. ## Name things right Talking about variables, there's one more extremely important thing. -Please name your variables sensibly. Take time to think about this. +A variable name should have a clean, obvious meaning, describing the data that it stores. -Variable naming is one of the most important and complex skills in programming. A quick glance at variable names can reveal which code was written by a beginner versus an experienced developer. +Variable naming is one of the most important and complex skills in programming. A glance at variable names can reveal which code was written by a beginner versus an experienced developer. -In a real project, most of the time is spent modifying and extending an existing code base rather than writing something completely separate from scratch. When we return to some code after doing something else for a while, it's much easier to find information that is well-labeled. Or, in other words, when the variables have good names. +In a real project, most of the time is spent modifying and extending an existing code base rather than writing something completely separate from scratch. When we return to some code after doing something else for a while, it's much easier to find information that is well-labelled. Or, in other words, when the variables have good names. Please spend time thinking about the right name for a variable before declaring it. Doing so will repay you handsomely. Some good-to-follow rules are: - Use human-readable names like `userName` or `shoppingCart`. -- Stay away from abbreviations or short names like `a`, `b`, `c`, unless you really know what you're doing. +- Stay away from abbreviations or short names like `a`, `b`, and `c`, unless you know what you're doing. - Make names maximally descriptive and concise. Examples of bad names are `data` and `value`. Such names say nothing. It's only okay to use them if the context of the code makes it exceptionally obvious which data or value the variable is referencing. -- Agree on terms within your team and in your own mind. If a site visitor is called a "user" then we should name related variables `currentUser` or `newUser` instead of `currentVisitor` or `newManInTown`. +- Agree on terms within your team and in your mind. If a site visitor is called a "user" then we should name related variables `currentUser` or `newUser` instead of `currentVisitor` or `newManInTown`. Sounds simple? Indeed it is, but creating descriptive and concise variable names in practice is not. Go for it. @@ -323,7 +338,7 @@ Modern JavaScript minifiers and browsers optimize code well enough, so it won't We can declare variables to store data by using the `var`, `let`, or `const` keywords. -- `let` -- is a modern variable declaration. The code must be in strict mode to use `let` in Chrome (V8). +- `let` -- is a modern variable declaration. - `var` -- is an old-school variable declaration. Normally we don't use it at all, but we'll cover subtle differences from `let` in the chapter <info:var>, just in case you need them. - `const` -- is like `let`, but the value of the variable can't be changed. diff --git a/1-js/02-first-steps/05-types/article.md b/1-js/02-first-steps/05-types/article.md index 0c9954ecf..04e8b2450 100644 --- a/1-js/02-first-steps/05-types/article.md +++ b/1-js/02-first-steps/05-types/article.md @@ -1,6 +1,10 @@ # Data types -A variable in JavaScript can contain any data. A variable can at one moment be a string and at another be a number: +A value in JavaScript is always of a certain type. For example, a string or a number. + +There are eight basic data types in JavaScript. Here, we'll cover them in general and in the next chapters we'll talk about each of them in detail. + +We can put any type in a variable. For example, a variable can at one moment be a string and then store a number: ```js // no error @@ -8,11 +12,9 @@ let message = "hello"; message = 123456; ``` -Programming languages that allow such things are called "dynamically typed", meaning that there are data types, but variables are not bound to any of them. +Programming languages that allow such things, such as JavaScript, are called "dynamically typed", meaning that there exist data types, but variables are not bound to any of them. -There are seven basic data types in JavaScript. Here, we'll cover them in general and in the next chapters we'll talk about each of them in detail. - -## A number +## Number ```js let n = 123; @@ -44,13 +46,15 @@ Besides regular numbers, there are so-called "special numeric values" which also alert( "not a number" / 2 ); // NaN, such division is erroneous ``` - `NaN` is sticky. Any further operation on `NaN` returns `NaN`: + `NaN` is sticky. Any further mathematical operation on `NaN` returns `NaN`: ```js run - alert( "not a number" / 2 + 5 ); // NaN + alert( NaN + 1 ); // NaN + alert( 3 * NaN ); // NaN + alert( "not a number" / 2 - 1 ); // NaN ``` - So, if there's a `NaN` somewhere in a mathematical expression, it propagates to the whole result. + So, if there's a `NaN` somewhere in a mathematical expression, it propagates to the whole result (there's only one exception to that: `NaN ** 0` is `1`). ```smart header="Mathematical operations are safe" Doing maths is "safe" in JavaScript. We can do anything: divide by zero, treat non-numeric strings as numbers, etc. @@ -62,14 +66,42 @@ Special numeric values formally belong to the "number" type. Of course they are We'll see more about working with numbers in the chapter <info:number>. -## A string +## BigInt [#bigint-type] + +In JavaScript, the "number" type cannot safely represent integer values larger than <code>(2<sup>53</sup>-1)</code> (that's `9007199254740991`), or less than <code>-(2<sup>53</sup>-1)</code> for negatives. + +To be really precise, the "number" type can store larger integers (up to <code>1.7976931348623157 * 10<sup>308</sup></code>), but outside of the safe integer range <code>±(2<sup>53</sup>-1)</code> there'll be a precision error, because not all digits fit into the fixed 64-bit storage. So an "approximate" value may be stored. + +For example, these two numbers (right above the safe range) are the same: + +```js +console.log(9007199254740991 + 1); // 9007199254740992 +console.log(9007199254740991 + 2); // 9007199254740992 +``` + +So to say, all odd integers greater than <code>(2<sup>53</sup>-1)</code> can't be stored at all in the "number" type. + +For most purposes <code>±(2<sup>53</sup>-1)</code> range is quite enough, but sometimes we need the entire range of really big integers, e.g. for cryptography or microsecond-precision timestamps. + +`BigInt` type was recently added to the language to represent integers of arbitrary length. + +A `BigInt` value is created by appending `n` to the end of an integer: + +```js +// the "n" at the end means it's a BigInt +const bigInt = 1234567890123456789012345678901234567890n; +``` + +As `BigInt` numbers are rarely needed, we don't cover them here, but devoted them a separate chapter <info:bigint>. Read it when you need such big numbers. + +## String A string in JavaScript must be surrounded by quotes. ```js let str = "Hello"; let str2 = 'Single quotes are ok too'; -let phrase = `can embed ${str}`; +let phrase = `can embed another ${str}`; ``` In JavaScript, there are 3 types of quotes. @@ -78,7 +110,7 @@ In JavaScript, there are 3 types of quotes. 2. Single quotes: `'Hello'`. 3. Backticks: <code>`Hello`</code>. -Double and single quotes are "simple" quotes. There's no difference between them in JavaScript. +Double and single quotes are "simple" quotes. There's practically no difference between them in JavaScript. Backticks are "extended functionality" quotes. They allow us to embed variables and expressions into a string by wrapping them in `${…}`, for example: @@ -102,12 +134,12 @@ alert( "the result is ${1 + 2}" ); // the result is ${1 + 2} (double quotes do n We'll cover strings more thoroughly in the chapter <info:string>. ```smart header="There is no *character* type." -In some languages, there is a special "character" type for a single character. For example, in the C language and in Java it is `char`. +In some languages, there is a special "character" type for a single character. For example, in the C language and in Java it is called "char". -In JavaScript, there is no such type. There's only one type: `string`. A string may consist of only one character or many of them. +In JavaScript, there is no such type. There's only one type: `string`. A string may consist of zero characters (be empty), one character or many of them. ``` -## A boolean (logical type) +## Boolean (logical type) The boolean type has only two values: `true` and `false`. @@ -144,7 +176,7 @@ In JavaScript, `null` is not a "reference to a non-existing object" or a "null p It's just a special value which represents "nothing", "empty" or "value unknown". -The code above states that `age` is unknown or empty for some reason. +The code above states that `age` is unknown. ## The "undefined" value @@ -155,49 +187,47 @@ The meaning of `undefined` is "value is not assigned". If a variable is declared, but not assigned, then its value is `undefined`: ```js run -let x; +let age; -alert(x); // shows "undefined" +alert(age); // shows "undefined" ``` -Technically, it is possible to assign `undefined` to any variable: +Technically, it is possible to explicitly assign `undefined` to a variable: ```js run -let x = 123; +let age = 100; -x = undefined; +// change the value to undefined +age = undefined; -alert(x); // "undefined" +alert(age); // "undefined" ``` -...But we don't recommend doing that. Normally, we use `null` to assign an "empty" or "unknown" value to a variable, and we use `undefined` for checks like seeing if a variable has been assigned. +...But we don't recommend doing that. Normally, one uses `null` to assign an "empty" or "unknown" value to a variable, while `undefined` is reserved as a default initial value for unassigned things. ## Objects and Symbols The `object` type is special. -All other types are called "primitive" because their values can contain only a single thing (be it a string or a number or whatever). In contrast, objects are used to store collections of data and more complex entities. We'll deal with them later in the chapter <info:object> after we learn more about primitives. - -The `symbol` type is used to create unique identifiers for objects. We have to mention it here for completeness, but it's better to study this type after objects. - -## The typeof operator [#type-typeof] +All other types are called "primitive" because their values can contain only a single thing (be it a string or a number or whatever). In contrast, objects are used to store collections of data and more complex entities. -The `typeof` operator returns the type of the argument. It's useful when we want to process values of different types differently or just want to do a quick check. +Being that important, objects deserve a special treatment. We'll deal with them later in the chapter <info:object>, after we learn more about primitives. -It supports two forms of syntax: +The `symbol` type is used to create unique identifiers for objects. We have to mention it here for the sake of completeness, but also postpone the details till we know objects. -1. As an operator: `typeof x`. -2. As a function: `typeof(x)`. +## The typeof operator [#type-typeof] -In other words, it works with parentheses or without them. The result is the same. +The `typeof` operator returns the type of the operand. It's useful when we want to process values of different types differently or just want to do a quick check. -The call to `typeof x` returns a string with the type name: +A call to `typeof x` returns a string with the type name: ```js typeof undefined // "undefined" typeof 0 // "number" +typeof 10n // "bigint" + typeof true // "boolean" typeof "foo" // "string" @@ -220,25 +250,37 @@ typeof alert // "function" (3) The last three lines may need additional explanation: 1. `Math` is a built-in object that provides mathematical operations. We will learn it in the chapter <info:number>. Here, it serves just as an example of an object. -2. The result of `typeof null` is `"object"`. That's wrong. It is an officially recognized error in `typeof`, kept for compatibility. Of course, `null` is not an object. It is a special value with a separate type of its own. So, again, this is an error in the language. -3. The result of `typeof alert` is `"function"`, because `alert` is a function of the language. We'll study functions in the next chapters where we'll see that there's no special "function" type in JavaScript. Functions belong to the object type. But `typeof` treats them differently. Formally, it's incorrect, but very convenient in practice. +2. The result of `typeof null` is `"object"`. That's an officially recognized error in `typeof`, coming from very early days of JavaScript and kept for compatibility. Definitely, `null` is not an object. It is a special value with a separate type of its own. The behavior of `typeof` is wrong here. +3. The result of `typeof alert` is `"function"`, because `alert` is a function. We'll study functions in the next chapters where we'll also see that there's no special "function" type in JavaScript. Functions belong to the object type. But `typeof` treats them differently, returning `"function"`. That also comes from the early days of JavaScript. Technically, such behavior isn't correct, but can be convenient in practice. + +```smart header="The `typeof(x)` syntax" +You may also come across another syntax: `typeof(x)`. It's the same as `typeof x`. +To put it clear: `typeof` is an operator, not a function. The parentheses here aren't a part of `typeof`. It's the kind of parentheses used for mathematical grouping. + +Usually, such parentheses contain a mathematical expression, such as `(2 + 2)`, but here they contain only one argument `(x)`. Syntactically, they allow to avoid a space between the `typeof` operator and its argument, and some people like it. + +Some people prefer `typeof(x)`, although the `typeof x` syntax is much more common. +``` ## Summary -There are 7 basic data types in JavaScript. +There are 8 basic data types in JavaScript. -- `number` for numbers of any kind: integer or floating-point. -- `string` for strings. A string may have one or more characters, there's no separate single-character type. -- `boolean` for `true`/`false`. -- `null` for unknown values -- a standalone type that has a single value `null`. -- `undefined` for unassigned values -- a standalone type that has a single value `undefined`. -- `object` for more complex data structures. -- `symbol` for unique identifiers. +- Seven primitive data types: + - `number` for numbers of any kind: integer or floating-point, integers are limited by <code>±(2<sup>53</sup>-1)</code>. + - `bigint` for integer numbers of arbitrary length. + - `string` for strings. A string may have zero or more characters, there's no separate single-character type. + - `boolean` for `true`/`false`. + - `null` for unknown values -- a standalone type that has a single value `null`. + - `undefined` for unassigned values -- a standalone type that has a single value `undefined`. + - `symbol` for unique identifiers. +- And one non-primitive data type: + - `object` for more complex data structures. The `typeof` operator allows us to see which type is stored in a variable. -- Two forms: `typeof x` or `typeof(x)`. +- Usually used as `typeof x`, but `typeof(x)` is also possible. - Returns a string with the name of the type, like `"string"`. - For `null` returns `"object"` -- this is an error in the language, it's not actually an object. diff --git a/1-js/02-first-steps/09-alert-prompt-confirm/1-simple-page/solution.md b/1-js/02-first-steps/06-alert-prompt-confirm/1-simple-page/solution.md similarity index 100% rename from 1-js/02-first-steps/09-alert-prompt-confirm/1-simple-page/solution.md rename to 1-js/02-first-steps/06-alert-prompt-confirm/1-simple-page/solution.md diff --git a/1-js/02-first-steps/09-alert-prompt-confirm/1-simple-page/task.md b/1-js/02-first-steps/06-alert-prompt-confirm/1-simple-page/task.md similarity index 100% rename from 1-js/02-first-steps/09-alert-prompt-confirm/1-simple-page/task.md rename to 1-js/02-first-steps/06-alert-prompt-confirm/1-simple-page/task.md diff --git a/1-js/02-first-steps/09-alert-prompt-confirm/article.md b/1-js/02-first-steps/06-alert-prompt-confirm/article.md similarity index 71% rename from 1-js/02-first-steps/09-alert-prompt-confirm/article.md rename to 1-js/02-first-steps/06-alert-prompt-confirm/article.md index f69c2d233..ef0f333cb 100644 --- a/1-js/02-first-steps/09-alert-prompt-confirm/article.md +++ b/1-js/02-first-steps/06-alert-prompt-confirm/article.md @@ -1,18 +1,10 @@ # Interaction: alert, prompt, confirm -This part of the tutorial aims to cover JavaScript "as is", without environment-specific tweaks. - -But we'll still be using the browser as our demo environment, so we should know at least a few of its user-interface functions. In this chapter, we'll get familiar with the browser functions `alert`, `prompt` and `confirm`. +As we'll be using the browser as our demo environment, let's see a couple of functions to interact with the user: `alert`, `prompt` and `confirm`. ## alert -Syntax: - -```js -alert(message); -``` - -This shows a message and pauses script execution until the user presses "OK". +This one we've seen already. It shows a message and waits for the user to press "OK". For example: @@ -20,7 +12,7 @@ For example: alert("Hello"); ``` -The mini-window with the message is called a *modal window*. The word "modal" means that the visitor can't interact with the rest of the page, press other buttons, etc. until they have dealt with the window. In this case -- until they press "OK". +The mini-window with the message is called a *modal window*. The word "modal" means that the visitor can't interact with the rest of the page, press other buttons, etc, until they have dealt with the window. In this case -- until they press "OK". ## prompt @@ -30,7 +22,7 @@ The function `prompt` accepts two arguments: result = prompt(title, [default]); ``` -It shows a modal window with a text message, an input field for the visitor, and the buttons OK/CANCEL. +It shows a modal window with a text message, an input field for the visitor, and the buttons OK/Cancel. `title` : The text to show the visitor. @@ -38,7 +30,11 @@ It shows a modal window with a text message, an input field for the visitor, and `default` : An optional second parameter, the initial value for the input field. -The visitor may type something in the prompt input field and press OK. Or they can cancel the input by pressing CANCEL or hitting the `key:Esc` key. +```smart header="The square brackets in syntax `[...]`" +The square brackets around `default` in the syntax above denote that the parameter is optional, not required. +``` + +The visitor can type something in the prompt input field and press OK. Then we get that text in the `result`. Or they can cancel the input by pressing Cancel or hitting the `key:Esc` key, then we get `null` as the `result`. The call to `prompt` returns the text from the input field or `null` if the input was canceled. @@ -74,7 +70,7 @@ The syntax: result = confirm(question); ``` -The function `confirm` shows a modal window with a `question` and two buttons: OK and CANCEL. +The function `confirm` shows a modal window with a `question` and two buttons: OK and Cancel. The result is `true` if OK is pressed and `false` otherwise. @@ -94,10 +90,10 @@ We covered 3 browser-specific functions to interact with visitors: : shows a message. `prompt` -: shows a message asking the user to input text. It returns the text or, if CANCEL or `key:Esc` is clicked, `null`. +: shows a message asking the user to input text. It returns the text or, if Cancel button or `key:Esc` is clicked, `null`. `confirm` -: shows a message and waits for the user to press "OK" or "CANCEL". It returns `true` for OK and `false` for CANCEL/`key:Esc`. +: shows a message and waits for the user to press "OK" or "Cancel". It returns `true` for OK and `false` for Cancel/`key:Esc`. All these methods are modal: they pause script execution and don't allow the visitor to interact with the rest of the page until the window has been dismissed. diff --git a/1-js/02-first-steps/06-type-conversions/article.md b/1-js/02-first-steps/07-type-conversions/article.md similarity index 70% rename from 1-js/02-first-steps/06-type-conversions/article.md rename to 1-js/02-first-steps/07-type-conversions/article.md index 6ac695e84..329556141 100644 --- a/1-js/02-first-steps/06-type-conversions/article.md +++ b/1-js/02-first-steps/07-type-conversions/article.md @@ -1,16 +1,18 @@ # Type Conversions -Most of the time, operators and functions automatically convert the values given to them to the right type. +Most of the time, operators and functions automatically convert the values given to them to the right type. For example, `alert` automatically converts any value to a string to show it. Mathematical operations convert values to numbers. There are also cases when we need to explicitly convert a value to the expected type. ```smart header="Not talking about objects yet" -In this chapter, we won't cover objects. Instead, we'll study primitives first. Later, after we learn about objects, we'll see how object conversion works in the chapter <info:object-toprimitive>. +In this chapter, we won't cover objects. For now, we'll just be talking about primitives. + +Later, after we learn about objects, in the chapter <info:object-toprimitive> we'll see how objects fit in. ``` -## ToString +## String Conversion String conversion happens when we need the string form of a value. @@ -30,9 +32,9 @@ alert(typeof value); // string String conversion is mostly obvious. A `false` becomes `"false"`, `null` becomes `"null"`, etc. -## ToNumber +## Numeric Conversion -Numeric conversion happens in mathematical functions and expressions automatically. +Numeric conversion in mathematical functions and expressions happens automatically. For example, when division `/` is applied to non-numbers: @@ -68,7 +70,7 @@ Numeric conversion rules: |`undefined`|`NaN`| |`null`|`0`| |<code>true and false</code> | `1` and `0` | -| `string` | Whitespaces from the start and end are removed. If the remaining string is empty, the result is `0`. Otherwise, the number is "read" from the string. An error gives `NaN`. | +| `string` | Whitespaces (includes spaces, tabs `\t`, newlines `\n` etc.) from the start and end are removed. If the remaining string is empty, the result is `0`. Otherwise, the number is "read" from the string. An error gives `NaN`. | Examples: @@ -81,20 +83,9 @@ alert( Number(false) ); // 0 Please note that `null` and `undefined` behave differently here: `null` becomes zero while `undefined` becomes `NaN`. -````smart header="Addition '+' concatenates strings" -Almost all mathematical operations convert values to numbers. A notable exception is addition `+`. If one of the added values is a string, the other one is also converted to a string. - -Then, it concatenates (joins) them: - -```js run -alert( 1 + '2' ); // '12' (string to the right) -alert( '1' + 2 ); // '12' (string to the left) -``` - -This only happens when at least one of the arguments is a string. Otherwise, values are converted to numbers. -```` +Most mathematical operators also perform such conversion, we'll see that in the next chapter. -## ToBoolean +## Boolean Conversion Boolean conversion is the simplest one. @@ -124,14 +115,13 @@ alert( Boolean(" ") ); // spaces, also true (any non-empty string is true) ``` ```` - ## Summary The three most widely used type conversions are to string, to number, and to boolean. -**`ToString`** -- Occurs when we output something. Can be performed with `String(value)`. The conversion to string is usually obvious for primitive values. +**`String Conversion`** -- Occurs when we output something. Can be performed with `String(value)`. The conversion to string is usually obvious for primitive values. -**`ToNumber`** -- Occurs in math operations. Can be performed with `Number(value)`. +**`Numeric Conversion`** -- Occurs in math operations. Can be performed with `Number(value)`. The conversion follows the rules: @@ -140,9 +130,9 @@ The conversion follows the rules: |`undefined`|`NaN`| |`null`|`0`| |<code>true / false</code> | `1 / 0` | -| `string` | The string is read "as is", whitespaces from both sides are ignored. An empty string becomes `0`. An error gives `NaN`. | +| `string` | The string is read "as is", whitespaces (includes spaces, tabs `\t`, newlines `\n` etc.) from both sides are ignored. An empty string becomes `0`. An error gives `NaN`. | -**`ToBoolean`** -- Occurs in logical operations. Can be performed with `Boolean(value)`. +**`Boolean Conversion`** -- Occurs in logical operations. Can be performed with `Boolean(value)`. Follows the rules: diff --git a/1-js/02-first-steps/08-comparison/1-comparison-questions/solution.md b/1-js/02-first-steps/08-comparison/1-comparison-questions/solution.md deleted file mode 100644 index 5c8bd2bc4..000000000 --- a/1-js/02-first-steps/08-comparison/1-comparison-questions/solution.md +++ /dev/null @@ -1,21 +0,0 @@ - - -```js no-beautify -5 > 4 → true -"apple" > "pineapple" → false -"2" > "12" → true -undefined == null → true -undefined === null → false -null == "\n0\n" → false -null === +"\n0\n" → false -``` - -Some of the reasons: - -1. Obviously, true. -2. Dictionary comparison, hence false. -3. Again, dictionary comparison, first char of `"2"` is greater than the first char of `"1"`. -4. Values `null` and `undefined` equal each other only. -5. Strict equality is strict. Different types from both sides lead to false. -6. See (4). -7. Strict equality of different types. diff --git a/1-js/02-first-steps/07-operators/1-increment-order/solution.md b/1-js/02-first-steps/08-operators/1-increment-order/solution.md similarity index 100% rename from 1-js/02-first-steps/07-operators/1-increment-order/solution.md rename to 1-js/02-first-steps/08-operators/1-increment-order/solution.md diff --git a/1-js/02-first-steps/07-operators/1-increment-order/task.md b/1-js/02-first-steps/08-operators/1-increment-order/task.md similarity index 100% rename from 1-js/02-first-steps/07-operators/1-increment-order/task.md rename to 1-js/02-first-steps/08-operators/1-increment-order/task.md diff --git a/1-js/02-first-steps/07-operators/2-assignment-result/solution.md b/1-js/02-first-steps/08-operators/2-assignment-result/solution.md similarity index 100% rename from 1-js/02-first-steps/07-operators/2-assignment-result/solution.md rename to 1-js/02-first-steps/08-operators/2-assignment-result/solution.md diff --git a/1-js/02-first-steps/07-operators/2-assignment-result/task.md b/1-js/02-first-steps/08-operators/2-assignment-result/task.md similarity index 100% rename from 1-js/02-first-steps/07-operators/2-assignment-result/task.md rename to 1-js/02-first-steps/08-operators/2-assignment-result/task.md diff --git a/1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/solution.md b/1-js/02-first-steps/08-operators/3-primitive-conversions-questions/solution.md similarity index 69% rename from 1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/solution.md rename to 1-js/02-first-steps/08-operators/3-primitive-conversions-questions/solution.md index 7dd0d61c2..7370b66af 100644 --- a/1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/solution.md +++ b/1-js/02-first-steps/08-operators/3-primitive-conversions-questions/solution.md @@ -9,11 +9,11 @@ true + false = 1 "$" + 4 + 5 = "$45" "4" - 2 = 2 "4px" - 2 = NaN -7 / 0 = Infinity -" -9 " + 5 = " -9 5" // (3) -" -9 " - 5 = -14 // (4) +" -9 " + 5 = " -9 5" // (3) +" -9 " - 5 = -14 // (4) null + 1 = 1 // (5) undefined + 1 = NaN // (6) +" \t \n" - 2 = -2 // (7) ``` 1. The addition with a string `"" + 1` converts `1` to a string: `"" + 1 = "1"`, and then we have `"1" + 0`, the same rule is applied. @@ -22,3 +22,4 @@ undefined + 1 = NaN // (6) 4. The subtraction always converts to numbers, so it makes `" -9 "` a number `-9` (ignoring spaces around it). 5. `null` becomes `0` after the numeric conversion. 6. `undefined` becomes `NaN` after the numeric conversion. +7. Space characters are trimmed off string start and end when a string is converted to a number. Here the whole string consists of space characters, such as `\t`, `\n` and a "regular" space between them. So, similarly to an empty string, it becomes `0`. diff --git a/1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/task.md b/1-js/02-first-steps/08-operators/3-primitive-conversions-questions/task.md similarity index 95% rename from 1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/task.md rename to 1-js/02-first-steps/08-operators/3-primitive-conversions-questions/task.md index f17e870de..068420c7d 100644 --- a/1-js/02-first-steps/06-type-conversions/1-primitive-conversions-questions/task.md +++ b/1-js/02-first-steps/08-operators/3-primitive-conversions-questions/task.md @@ -16,11 +16,11 @@ true + false "$" + 4 + 5 "4" - 2 "4px" - 2 -7 / 0 " -9 " + 5 " -9 " - 5 null + 1 undefined + 1 +" \t \n" - 2 ``` Think well, write down and then compare with the answer. diff --git a/1-js/02-first-steps/08-operators/4-fix-prompt/solution.md b/1-js/02-first-steps/08-operators/4-fix-prompt/solution.md new file mode 100644 index 000000000..209a0702c --- /dev/null +++ b/1-js/02-first-steps/08-operators/4-fix-prompt/solution.md @@ -0,0 +1,32 @@ +The reason is that prompt returns user input as a string. + +So variables have values `"1"` and `"2"` respectively. + +```js run +let a = "1"; // prompt("First number?", 1); +let b = "2"; // prompt("Second number?", 2); + +alert(a + b); // 12 +``` + +What we should do is to convert strings to numbers before `+`. For example, using `Number()` or prepending them with `+`. + +For example, right before `prompt`: + +```js run +let a = +prompt("First number?", 1); +let b = +prompt("Second number?", 2); + +alert(a + b); // 3 +``` + +Or in the `alert`: + +```js run +let a = prompt("First number?", 1); +let b = prompt("Second number?", 2); + +alert(+a + +b); // 3 +``` + +Using both unary and binary `+` in the latest code. Looks funny, doesn't it? diff --git a/1-js/02-first-steps/08-operators/4-fix-prompt/task.md b/1-js/02-first-steps/08-operators/4-fix-prompt/task.md new file mode 100644 index 000000000..b3ea4a3a3 --- /dev/null +++ b/1-js/02-first-steps/08-operators/4-fix-prompt/task.md @@ -0,0 +1,18 @@ +importance: 5 + +--- + +# Fix the addition + +Here's a code that asks the user for two numbers and shows their sum. + +It works incorrectly. The output in the example below is `12` (for default prompt values). + +Why? Fix it. The result should be `3`. + +```js run +let a = prompt("First number?", 1); +let b = prompt("Second number?", 2); + +alert(a + b); // 12 +``` diff --git a/1-js/02-first-steps/07-operators/article.md b/1-js/02-first-steps/08-operators/article.md similarity index 67% rename from 1-js/02-first-steps/07-operators/article.md rename to 1-js/02-first-steps/08-operators/article.md index 74b27e871..d52c37a17 100644 --- a/1-js/02-first-steps/07-operators/article.md +++ b/1-js/02-first-steps/08-operators/article.md @@ -1,8 +1,8 @@ -# Operators +# Basic operators, maths We know many operators from school. They are things like addition `+`, multiplication `*`, subtraction `-`, and so on. -In this chapter, we'll concentrate on aspects of operators that are not covered by school arithmetic. +In this chapter, we’ll start with simple operators, then concentrate on JavaScript-specific aspects, not covered by school arithmetic. ## Terms: "unary", "binary", "operand" @@ -26,11 +26,62 @@ Before we move on, let's grasp some common terminology. alert( y - x ); // 2, binary minus subtracts values ``` - Formally, we're talking about two different operators here: the unary negation (single operand: reverses the sign) and the binary subtraction (two operands: subtracts). + Formally, in the examples above we have two different operators that share the same symbol: the negation operator, a unary operator that reverses the sign, and the subtraction operator, a binary operator that subtracts one number from another. -## String concatenation, binary + +## Maths -Now, let's see special features of JavaScript operators that are beyond school arithmetics. +The following math operations are supported: + +- Addition `+`, +- Subtraction `-`, +- Multiplication `*`, +- Division `/`, +- Remainder `%`, +- Exponentiation `**`. + +The first four are straightforward, while `%` and `**` need a few words about them. + +### Remainder % + +The remainder operator `%`, despite its appearance, is not related to percents. + +The result of `a % b` is the [remainder](https://en.wikipedia.org/wiki/Remainder) of the integer division of `a` by `b`. + +For instance: + +```js run +alert( 5 % 2 ); // 1, the remainder of 5 divided by 2 +alert( 8 % 3 ); // 2, the remainder of 8 divided by 3 +alert( 8 % 4 ); // 0, the remainder of 8 divided by 4 +``` + +### Exponentiation ** + +The exponentiation operator `a ** b` raises `a` to the power of `b`. + +In school maths, we write that as a<sup>b</sup>. + +For instance: + +```js run +alert( 2 ** 2 ); // 2² = 4 +alert( 2 ** 3 ); // 2³ = 8 +alert( 2 ** 4 ); // 2⁴ = 16 +``` + +Just like in maths, the exponentiation operator is defined for non-integer numbers as well. + +For example, a square root is an exponentiation by ½: + +```js run +alert( 4 ** (1/2) ); // 2 (power of 1/2 is the same as a square root) +alert( 8 ** (1/3) ); // 2 (power of 1/3 is the same as a cubic root) +``` + + +## String concatenation with binary + + +Let's meet the features of JavaScript operators that are beyond school arithmetics. Usually, the plus operator `+` sums numbers. @@ -41,7 +92,7 @@ let s = "my" + "string"; alert(s); // mystring ``` -Note that if one of the operands is a string, the other one is converted to a string too. +Note that if any of the operands is a string, then the other one is converted to a string too. For example: @@ -50,22 +101,28 @@ alert( '1' + 2 ); // "12" alert( 2 + '1' ); // "21" ``` -See, it doesn't matter whether the first operand is a string or the second one. The rule is simple: if either operand is a string, the other one is converted into a string as well. - -However, note that operations run from left to right. If there are two numbers followed by a string, the numbers will be added before being converted to a string: +See, it doesn't matter whether the first operand is a string or the second one. +Here's a more complex example: ```js run alert(2 + 2 + '1' ); // "41" and not "221" ``` -String concatenation and conversion is a special feature of the binary plus `+`. Other arithmetic operators work only with numbers and always convert their operands to numbers. +Here, operators work one after another. The first `+` sums two numbers, so it returns `4`, then the next `+` adds the string `1` to it, so it's like `4 + '1' = '41'`. -For instance, subtraction and division: +```js run +alert('1' + 2 + 2); // "122" and not "14" +``` +Here, the first operand is a string, the compiler treats the other two operands as strings too. The `2` gets concatenated to `'1'`, so it's like `'1' + 2 = "12"` and `"12" + 2 = "122"`. + +The binary `+` is the only operator that supports strings in such a way. Other arithmetic operators work only with numbers and always convert their operands to numbers. + +Here's the demo for subtraction and division: ```js run -alert( 2 - '1' ); // 1 -alert( '6' / '2' ); // 3 +alert( 6 - '2' ); // 4, converts '2' to a number +alert( '6' / '2' ); // 3, converts both operands to numbers ``` ## Numeric conversion, unary + @@ -93,9 +150,7 @@ alert( +"" ); // 0 It actually does the same thing as `Number(...)`, but is shorter. -The need to convert strings to numbers arises very often. For example, if we are getting values from HTML form fields, they are usually strings. - -What if we want to sum them? +The need to convert strings to numbers arises very often. For example, if we are getting values from HTML form fields, they are usually strings. What if we want to sum them? The binary plus would add them as strings: @@ -127,34 +182,35 @@ Why are unary pluses applied to values before the binary ones? As we're going to ## Operator precedence -If an expression has more than one operator, the execution order is defined by their *precedence*, or, in other words, the implicit priority order of operators. +If an expression has more than one operator, the execution order is defined by their *precedence*, or, in other words, the default priority order of operators. From school, we all know that the multiplication in the expression `1 + 2 * 2` should be calculated before the addition. That's exactly the precedence thing. The multiplication is said to have *a higher precedence* than the addition. -Parentheses override any precedence, so if we're not satisfied with the implicit order, we can use them to change it. For example: `(1 + 2) * 2`. +Parentheses override any precedence, so if we're not satisfied with the default order, we can use them to change it. For example, write `(1 + 2) * 2`. There are many operators in JavaScript. Every operator has a corresponding precedence number. The one with the larger number executes first. If the precedence is the same, the execution order is from left to right. -Here's an extract from the [precedence table](https://developer.mozilla.org/en/JavaScript/Reference/operators/operator_precedence) (you don't need to remember this, but note that unary operators are higher than corresponding binary ones): +Here's an extract from the [precedence table](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Operator_Precedence) (you don't need to remember this, but note that unary operators are higher than corresponding binary ones): | Precedence | Name | Sign | |------------|------|------| | ... | ... | ... | -| 16 | unary plus | `+` | -| 16 | unary negation | `-` | -| 14 | multiplication | `*` | -| 14 | division | `/` | -| 13 | addition | `+` | -| 13 | subtraction | `-` | +| 14 | unary plus | `+` | +| 14 | unary negation | `-` | +| 13 | exponentiation | `**` | +| 12 | multiplication | `*` | +| 12 | division | `/` | +| 11 | addition | `+` | +| 11 | subtraction | `-` | | ... | ... | ... | -| 3 | assignment | `=` | +| 2 | assignment | `=` | | ... | ... | ... | -As we can see, the "unary plus" has a priority of `16` which is higher than the `13` of "addition" (binary plus). That's why, in the expression `"+apples + +oranges"`, unary pluses work before the addition. +As we can see, the "unary plus" has a priority of `14` which is higher than the `11` of "addition" (binary plus). That's why, in the expression `"+apples + +oranges"`, unary pluses work before the addition. ## Assignment -Let's note that an assignment `=` is also an operator. It is listed in the precedence table with the very low priority of `3`. +Let's note that an assignment `=` is also an operator. It is listed in the precedence table with the very low priority of `2`. That's why, when we assign a variable, like `x = 2 * 2 + 1`, the calculations are done first and then the `=` is evaluated, storing the result in `x`. @@ -164,24 +220,11 @@ let x = 2 * 2 + 1; alert( x ); // 5 ``` -It is possible to chain assignments: - -```js run -let a, b, c; - -*!* -a = b = c = 2 + 2; -*/!* - -alert( a ); // 4 -alert( b ); // 4 -alert( c ); // 4 -``` +### Assignment = returns a value -Chained assignments evaluate from right to left. First, the rightmost expression `2 + 2` is evaluated and then assigned to the variables on the left: `c`, `b` and `a`. At the end, all the variables share a single value. +The fact of `=` being an operator, not a "magical" language construct has an interesting implication. -````smart header="The assignment operator `\"=\"` returns a value" -An operator always returns a value. That's obvious for most of them like addition `+` or multiplication `*`. But the assignment operator follows this rule too. +All operators in JavaScript return a value. That's obvious for `+` and `-`, but also true for `=`. The call `x = value` writes the `value` into `x` *and then returns it*. @@ -199,51 +242,76 @@ alert( a ); // 3 alert( c ); // 0 ``` -In the example above, the result of `(a = b + 1)` is the value which is assigned to `a` (that is `3`). It is then used to subtract from `3`. - -Funny code, isn't it? We should understand how it works, because sometimes we see it in 3rd-party libraries, but shouldn't write anything like that ourselves. Such tricks definitely don't make code clearer or readable. -```` +In the example above, the result of expression `(a = b + 1)` is the value which was assigned to `a` (that is `3`). It is then used for further evaluations. -## Remainder % +Funny code, isn't it? We should understand how it works, because sometimes we see it in JavaScript libraries. -The remainder operator `%`, despite its appearance, is not related to percents. +Although, please don't write the code like that. Such tricks definitely don't make code clearer or readable. -The result of `a % b` is the remainder of the integer division of `a` by `b`. +### Chaining assignments -For instance: +Another interesting feature is the ability to chain assignments: ```js run -alert( 5 % 2 ); // 1 is a remainder of 5 divided by 2 -alert( 8 % 3 ); // 2 is a remainder of 8 divided by 3 -alert( 6 % 3 ); // 0 is a remainder of 6 divided by 3 +let a, b, c; + +*!* +a = b = c = 2 + 2; +*/!* + +alert( a ); // 4 +alert( b ); // 4 +alert( c ); // 4 ``` -## Exponentiation ** +Chained assignments evaluate from right to left. First, the rightmost expression `2 + 2` is evaluated and then assigned to the variables on the left: `c`, `b` and `a`. At the end, all the variables share a single value. -The exponentiation operator `**` is a recent addition to the language. +Once again, for the purposes of readability it's better to split such code into few lines: -For a natural number `b`, the result of `a ** b` is `a` multiplied by itself `b` times. +```js +c = 2 + 2; +b = c; +a = c; +``` +That's easier to read, especially when eye-scanning the code fast. -For instance: +## Modify-in-place + +We often need to apply an operator to a variable and store the new result in that same variable. + +For example: + +```js +let n = 2; +n = n + 5; +n = n * 2; +``` + +This notation can be shortened using the operators `+=` and `*=`: ```js run -alert( 2 ** 2 ); // 4 (2 * 2) -alert( 2 ** 3 ); // 8 (2 * 2 * 2) -alert( 2 ** 4 ); // 16 (2 * 2 * 2 * 2) +let n = 2; +n += 5; // now n = 7 (same as n = n + 5) +n *= 2; // now n = 14 (same as n = n * 2) + +alert( n ); // 14 ``` -The operator works for non-integer numbers as well. +Short "modify-and-assign" operators exist for all arithmetical and bitwise operators: `/=`, `-=`, etc. -For instance: +Such operators have the same precedence as a normal assignment, so they run after most other calculations: ```js run -alert( 4 ** (1/2) ); // 2 (power of 1/2 is the same as a square root, that's maths) -alert( 8 ** (1/3) ); // 2 (power of 1/3 is the same as a cubic root) +let n = 2; + +n *= 3 + 5; // right part evaluated first, same as n *= 8 + +alert( n ); // 16 ``` ## Increment/decrement -<!-- Can't use -- in title, because built-in parse turns it into – --> +<!-- Can't use -- in title, because the built-in parser turns it into a 'long dash' – --> Increasing or decreasing a number by one is among the most common numerical operations. @@ -253,14 +321,14 @@ So, there are special operators for it: ```js run no-beautify let counter = 2; - counter++; // works the same as counter = counter + 1, but is shorter + counter++; // works the same as counter = counter + 1, but is shorter alert( counter ); // 3 ``` - **Decrement** `--` decreases a variable by 1: ```js run no-beautify let counter = 2; - counter--; // works the same as counter = counter - 1, but is shorter + counter--; // works the same as counter = counter - 1, but is shorter alert( counter ); // 1 ``` @@ -370,41 +438,7 @@ The list of operators: - RIGHT SHIFT ( `>>` ) - ZERO-FILL RIGHT SHIFT ( `>>>` ) -These operators are used very rarely. To understand them, we need to delve into low-level number representation and it would not be optimal to do that right now, especially since we won't need them any time soon. If you're curious, you can read the [Bitwise Operators](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Operators/Bitwise_Operators) article on MDN. It would be more practical to do that when a real need arises. - -## Modify-in-place - -We often need to apply an operator to a variable and store the new result in that same variable. - -For example: - -```js -let n = 2; -n = n + 5; -n = n * 2; -``` - -This notation can be shortened using the operators `+=` and `*=`: - -```js run -let n = 2; -n += 5; // now n = 7 (same as n = n + 5) -n *= 2; // now n = 14 (same as n = n * 2) - -alert( n ); // 14 -``` - -Short "modify-and-assign" operators exist for all arithmetical and bitwise operators: `/=`, `-=`, etc. - -Such operators have the same precedence as a normal assignment, so they run after most other calculations: - -```js run -let n = 2; - -n *= 3 + 5; - -alert( n ); // 16 (right part evaluated first, same as n *= 8) -``` +These operators are used very rarely, when we need to fiddle with numbers on the very lowest (bitwise) level. We won't need these operators any time soon, as web development has little use of them, but in some special areas, such as cryptography, they are useful. You can read the [Bitwise Operators](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Expressions_and_Operators#bitwise_operators) chapter on MDN when a need arises. ## Comma @@ -427,10 +461,10 @@ Here, the first expression `1 + 2` is evaluated and its result is thrown away. T ```smart header="Comma has a very low precedence" Please note that the comma operator has very low precedence, lower than `=`, so parentheses are important in the example above. -Without them: `a = 1 + 2, 3 + 4` evaluates `+` first, summing the numbers into `a = 3, 7`, then the assignment operator `=` assigns `a = 3`, and finally the number after the comma, `7`, is not processed so it's ignored. +Without them: `a = 1 + 2, 3 + 4` evaluates `+` first, summing the numbers into `a = 3, 7`, then the assignment operator `=` assigns `a = 3`, and the rest is ignored. It's like `(a = 1 + 2), 3 + 4`. ``` -Why do we need an operator that throws away everything except the last part? +Why do we need an operator that throws away everything except the last expression? Sometimes, people use it in more complex constructs to put several actions in one line. @@ -443,4 +477,4 @@ for (*!*a = 1, b = 3, c = a * b*/!*; a < 10; a++) { } ``` -Such tricks are used in many JavaScript frameworks. That's why we're mentioning them. But, usually, they don't improve code readability so we should think well before using them. +Such tricks are used in many JavaScript frameworks. That's why we're mentioning them. But usually they don't improve code readability so we should think well before using them. diff --git a/1-js/02-first-steps/09-comparison/1-comparison-questions/solution.md b/1-js/02-first-steps/09-comparison/1-comparison-questions/solution.md new file mode 100644 index 000000000..632b1cf4e --- /dev/null +++ b/1-js/02-first-steps/09-comparison/1-comparison-questions/solution.md @@ -0,0 +1,21 @@ + + +```js no-beautify +5 > 4 → true +"apple" > "pineapple" → false +"2" > "12" → true +undefined == null → true +undefined === null → false +null == "\n0\n" → false +null === +"\n0\n" → false +``` + +Some of the reasons: + +1. Obviously, true. +2. Dictionary comparison, hence false. `"a"` is smaller than `"p"`. +3. Again, dictionary comparison, first char `"2"` is greater than the first char `"1"`. +4. Values `null` and `undefined` equal each other only. +5. Strict equality is strict. Different types from both sides lead to false. +6. Similar to `(4)`, `null` only equals `undefined`. +7. Strict equality of different types. diff --git a/1-js/02-first-steps/08-comparison/1-comparison-questions/task.md b/1-js/02-first-steps/09-comparison/1-comparison-questions/task.md similarity index 100% rename from 1-js/02-first-steps/08-comparison/1-comparison-questions/task.md rename to 1-js/02-first-steps/09-comparison/1-comparison-questions/task.md diff --git a/1-js/02-first-steps/08-comparison/article.md b/1-js/02-first-steps/09-comparison/article.md similarity index 84% rename from 1-js/02-first-steps/08-comparison/article.md rename to 1-js/02-first-steps/09-comparison/article.md index 3d5cc1729..a69317fee 100644 --- a/1-js/02-first-steps/08-comparison/article.md +++ b/1-js/02-first-steps/09-comparison/article.md @@ -1,15 +1,21 @@ # Comparisons -We know many comparison operators from maths: +We know many comparison operators from maths. + +In JavaScript they are written like this: - Greater/less than: <code>a > b</code>, <code>a < b</code>. - Greater/less than or equals: <code>a >= b</code>, <code>a <= b</code>. -- Equals: `a == b` (please note the double equals sign `=`. A single symbol `a = b` would mean an assignment). -- Not equals. In maths the notation is <code>≠</code>, but in JavaScript it's written as an assignment with an exclamation sign before it: <code>a != b</code>. +- Equals: `a == b`, please note the double equality sign `==` means the equality test, while a single one `a = b` means an assignment. +- Not equals: In maths the notation is <code>≠</code>, but in JavaScript it's written as <code>a != b</code>. + +In this article we'll learn more about different types of comparisons, how JavaScript makes them, including important peculiarities. + +At the end you'll find a good recipe to avoid "JavaScript quirks"-related issues. ## Boolean is the result -Like all other operators, a comparison returns a value. In this case, the value is a boolean. +All comparison operators return a boolean value: - `true` -- means "yes", "correct" or "the truth". - `false` -- means "no", "wrong" or "not the truth". @@ -51,7 +57,9 @@ The algorithm to compare two strings is simple: 4. Repeat until the end of either string. 5. If both strings end at the same length, then they are equal. Otherwise, the longer string is greater. -In the examples above, the comparison `'Z' > 'A'` gets to a result at the first step while the strings `"Glow"` and `"Glee"` are compared character-by-character: +In the first example above, the comparison `'Z' > 'A'` gets to a result at the first step. + +The second comparison `'Glow'` and `'Glee'` needs more steps as strings are compared character-by-character: 1. `G` is the same as `G`. 2. `l` is the same as `l`. @@ -74,7 +82,7 @@ alert( '2' > 1 ); // true, string '2' becomes a number 2 alert( '01' == 1 ); // true, string '01' becomes a number 1 ``` -For boolean values, `true` becomes `1` and `false` becomes `0`. +For boolean values, `true` becomes `1` and `false` becomes `0`. For example: @@ -138,11 +146,8 @@ The strict equality operator is a bit longer to write, but makes it obvious what ## Comparison with null and undefined -Let's see more edge cases. - There's a non-intuitive behavior when `null` or `undefined` are compared to other values. - For a strict equality check `===` : These values are different, because each of them is a different type. @@ -193,15 +198,14 @@ Why does it dislike zero so much? Always false! We get these results because: - Comparisons `(1)` and `(2)` return `false` because `undefined` gets converted to `NaN` and `NaN` is a special numeric value which returns `false` for all comparisons. -- The equality check `(3)` returns `false` because `undefined` only equals `null` and no other value. - -### Evade problems +- The equality check `(3)` returns `false` because `undefined` only equals `null`, `undefined`, and no other value. -Why did we go over these examples? Should we remember these peculiarities all the time? Well, not really. Actually, these tricky things will gradually become familiar over time, but there's a solid way to evade problems with them: +### Avoid problems -Just treat any comparison with `undefined/null` except the strict equality `===` with exceptional care. +Why did we go over these examples? Should we remember these peculiarities all the time? Well, not really. Actually, these tricky things will gradually become familiar over time, but there's a solid way to avoid problems with them: -Don't use comparisons `>= > < <=` with a variable which may be `null/undefined`, unless you're really sure of what you're doing. If a variable can have these values, check for them separately. +- Treat any comparison with `undefined/null` except the strict equality `===` with exceptional care. +- Don't use comparisons `>= > < <=` with a variable which may be `null/undefined`, unless you're really sure of what you're doing. If a variable can have these values, check for them separately. ## Summary diff --git a/1-js/02-first-steps/10-ifelse/2-check-standard/task.md b/1-js/02-first-steps/10-ifelse/2-check-standard/task.md index 418ef4a6d..f29db4b24 100644 --- a/1-js/02-first-steps/10-ifelse/2-check-standard/task.md +++ b/1-js/02-first-steps/10-ifelse/2-check-standard/task.md @@ -6,7 +6,11 @@ viktighet: 2 Ved å bruke `if..else` skal du skrive kode som spør: 'Hva er det "offisielle" navnet til JavaScript' +<<<<<<< HEAD Hvis brukeren skriver inn "ECMAScript", skal tilbakemeldingen være "Korrekt!". Ellers skal den være: "Visste du det ikke? ECMAScript!" +======= +If the visitor enters "ECMAScript", then output "Right!", otherwise -- output: "You don't know? ECMAScript!" +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  diff --git a/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/solution.md b/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/solution.md index 6b8d9bd67..b7c628174 100644 --- a/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/solution.md +++ b/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/solution.md @@ -1,3 +1,7 @@ ```js +<<<<<<< HEAD result = a + b < 4 ? "Below" : "Over"; +======= +let result = (a + b < 4) ? 'Below' : 'Over'; +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ``` diff --git a/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/task.md b/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/task.md index 421c7110f..2a666870b 100644 --- a/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/task.md +++ b/1-js/02-first-steps/10-ifelse/5-rewrite-if-question/task.md @@ -4,9 +4,15 @@ importance: 5 # Gjør om 'if' til '?' +<<<<<<< HEAD Gjør om denne `if` påstanden ved bruk av ternary operatøren`'?'`: +======= +Rewrite this `if` using the conditional operator `'?'`: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js +let result; + if (a + b < 4) { result = "Below"; } else { diff --git a/1-js/02-first-steps/10-ifelse/article.md b/1-js/02-first-steps/10-ifelse/article.md index 0b90d6364..0edca8a6a 100644 --- a/1-js/02-first-steps/10-ifelse/article.md +++ b/1-js/02-first-steps/10-ifelse/article.md @@ -1,12 +1,24 @@ +<<<<<<< HEAD # Operant betinging: if, '?' +======= +# Conditional branching: if, '?' +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Ofte i koden vår ønsker vi å utføre forskjellige operasjoner basert på ulike betingelser. +<<<<<<< HEAD For å bestemme hvilken kode som skal bli kjørt basert disse ulike betingelsene, bruker vi noe som kalles for "if" påstand (eng: if-statement). Man kan også benytte seg av en (ternary) operator, som vi vil skrive mer om etterpå. +======= +To do that, we can use the `if` statement and the conditional operator `?`, that's also called a "question mark" operator. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## "if" påstand +<<<<<<< HEAD Oppgaven til en `if` påstand er å evaluere en betingelse (condition). Om resultatet er sant (`true`) kjøres koden. +======= +The `if(...)` statement evaluates a condition in parentheses and, if the result is `true`, executes a block of code. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Eksempel: @@ -66,9 +78,13 @@ if (cond) { ## Else "leddsetningen" +<<<<<<< HEAD Det hender at man ofte vil kjøre én spesifikk kode hvis betingelsen i "if" påstanden stemmer, og ellers en annen kode hvis den ikke stemmer. For å gjøre dette bruker man noe som kalles for en "else statement". Denne kjøres uansett hvis "if" påstanden er false. Eksempel: +======= +The `if` statement may contain an optional `else` block. It executes when the condition is falsy. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js run let year = prompt( @@ -108,7 +124,11 @@ Fremgangsmåten til JavaScript er å først sjekke om `alder == 18)`. Hvis det e Merk: Det kan være flere `else if` påstander, men bare en `else` påstand. +<<<<<<< HEAD ## Ternary operatør '?' +======= +## Conditional operator '?' +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Det hender at man ønsker å sette en variabel basert på en betingelse. @@ -129,11 +149,17 @@ if (age > 18) { alert(accessAllowed); ``` +<<<<<<< HEAD For å gjøre dette på en enklere måte, kan vi bruke noe som heter "ternary operator". Noen kaller denne også for "question mark operator". Dettte er fordi denne operatøren brukes med et `?`-tegn. "Ternary" betyr at operatøren har tre "operander", som er tegnene som brukes i uttrykket. Denne er faktisk den eneste operatøren i JavaScript som har tre operander. Syntaksen er som følger: +======= +The so-called "conditional" or "question mark" operator lets us do that in a shorter and simpler way. + +The operator is represented by a question mark `?`. Sometimes it's called "ternary", because the operator has three operands. It is actually the one and only operator in JavaScript which has that many. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js let result = condition ? value1 : value2; @@ -147,7 +173,11 @@ For eksempel: let accessAllowed = age > 18 ? true : false; ``` +<<<<<<< HEAD Teknisk sett kan vi sette paranteser rundt `alder > 18`. Men spørsmålstegnet i en ternary operatør har en lav rang, og vil dermed bli kjørt etter `>`-sjekken. +======= +Technically, we can omit the parentheses around `age > 18`. The question mark operator has a low precedence, so it executes after the comparison `>`. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Dette eksemplet vil fungere likt som det forrige: @@ -191,10 +221,17 @@ alert(message); Det kan vøre uvant å forstå hva som foregår her med en gang. Tar vi en nærmere kikk derimot, ser vi at det bare er flere sekvenser med betingelser: +<<<<<<< HEAD 1. Det første spørsmålstegnet sjekker om `age < 3`. 2. Om det er "true" returnerer den `'Hi, baby!'`. Hvis den ikke er "true", går den til neste verdien etter kolontegenet '":"', som er `age < 18`. 3. Om det "true", returner den `'Hello!'`. Hvis den ikke er "true", går den til neste verdien etter kolontegenet '":"', som er `age < 100`. 4. Om det "true", returner den `'Greetings!'`. Hvis den ikke er "true", går den til neste verdien etter kolontegenet '":"', som er `'What an unusual age!'`. +======= +1. The first question mark checks whether `age < 3`. +2. If true -- it returns `'Hi, baby!'`. Otherwise, it continues to the expression after the colon ":", checking `age < 18`. +3. If that's true -- it returns `'Hello!'`. Otherwise, it continues to the expression after the next colon ":", checking `age < 100`. +4. If that's true -- it returns `'Greetings!'`. Otherwise, it continues to the expression after the last colon ":", returning `'What an unusual age!'`. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Her kan du se hvordan det ville sett ut med `if..else`: @@ -227,7 +264,11 @@ Avhengig om påstanden `selskap == 'Netscape` stemmer eller ikke, vil enten den Her satt vi ikke resultatet til en variabel, men kjørte forskjellig kode avhengig av påstanden. +<<<<<<< HEAD **Vi anbefaler ikke at man bruker spørsmålstegnet på denne måten.** +======= +**It's not recommended to use the question mark operator in this way.** +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Å skrive det på denne måten er kortere enn en `if` påstand, som kan være attraktivt for mange programmerere. Men det er vanskeligere å lese, og sees dermed på som dårlig vane. diff --git a/1-js/02-first-steps/11-logical-operators/2-alert-or/solution.md b/1-js/02-first-steps/11-logical-operators/2-alert-or/solution.md index 8f4d664e8..f85b56366 100644 --- a/1-js/02-first-steps/11-logical-operators/2-alert-or/solution.md +++ b/1-js/02-first-steps/11-logical-operators/2-alert-or/solution.md @@ -6,7 +6,7 @@ alert( alert(1) || 2 || alert(3) ); The call to `alert` does not return a value. Or, in other words, it returns `undefined`. -1. The first OR `||` evaluates it's left operand `alert(1)`. That shows the first message with `1`. +1. The first OR `||` evaluates its left operand `alert(1)`. That shows the first message with `1`. 2. The `alert` returns `undefined`, so OR goes on to the second operand searching for a truthy value. 3. The second operand `2` is truthy, so the execution is halted, `2` is returned and then shown by the outer alert. diff --git a/1-js/02-first-steps/11-logical-operators/3-alert-1-null-2/solution.md b/1-js/02-first-steps/11-logical-operators/3-alert-1-null-2/solution.md index 5c2455ef4..368b59409 100644 --- a/1-js/02-first-steps/11-logical-operators/3-alert-1-null-2/solution.md +++ b/1-js/02-first-steps/11-logical-operators/3-alert-1-null-2/solution.md @@ -1,6 +1,6 @@ The answer: `null`, because it's the first falsy value from the list. ```js run -alert( 1 && null && 2 ); +alert(1 && null && 2); ``` diff --git a/1-js/02-first-steps/11-logical-operators/6-check-if-in-range/task.md b/1-js/02-first-steps/11-logical-operators/6-check-if-in-range/task.md index cc00ca9fc..fc9e336c1 100644 --- a/1-js/02-first-steps/11-logical-operators/6-check-if-in-range/task.md +++ b/1-js/02-first-steps/11-logical-operators/6-check-if-in-range/task.md @@ -4,6 +4,6 @@ importance: 3 # Check the range between -Write an "if" condition to check that `age` is between `14` and `90` inclusively. +Write an `if` condition to check that `age` is between `14` and `90` inclusively. "Inclusively" means that `age` can reach the edges `14` or `90`. diff --git a/1-js/02-first-steps/11-logical-operators/7-check-if-out-range/task.md b/1-js/02-first-steps/11-logical-operators/7-check-if-out-range/task.md index 7c22d6ad1..9b947d00f 100644 --- a/1-js/02-first-steps/11-logical-operators/7-check-if-out-range/task.md +++ b/1-js/02-first-steps/11-logical-operators/7-check-if-out-range/task.md @@ -4,6 +4,6 @@ importance: 3 # Check the range outside -Write an `if` condition to check that `age` is NOT between 14 and 90 inclusively. +Write an `if` condition to check that `age` is NOT between `14` and `90` inclusively. Create two variants: the first one using NOT `!`, the second one -- without it. diff --git a/1-js/02-first-steps/11-logical-operators/9-check-login/solution.md b/1-js/02-first-steps/11-logical-operators/9-check-login/solution.md index b535650ec..604606259 100644 --- a/1-js/02-first-steps/11-logical-operators/9-check-login/solution.md +++ b/1-js/02-first-steps/11-logical-operators/9-check-login/solution.md @@ -3,19 +3,19 @@ ```js run demo let userName = prompt("Who's there?", ''); -if (userName == 'Admin') { +if (userName === 'Admin') { let pass = prompt('Password?', ''); - if (pass == 'TheMaster') { + if (pass === 'TheMaster') { alert( 'Welcome!' ); - } else if (pass == '' || pass == null) { - alert( 'Canceled.' ); + } else if (pass === '' || pass === null) { + alert( 'Canceled' ); } else { alert( 'Wrong password' ); } -} else if (userName == '' || userName == null) { +} else if (userName === '' || userName === null) { alert( 'Canceled' ); } else { alert( "I don't know you" ); diff --git a/1-js/02-first-steps/11-logical-operators/9-check-login/task.md b/1-js/02-first-steps/11-logical-operators/9-check-login/task.md index 0728efad1..290a52642 100644 --- a/1-js/02-first-steps/11-logical-operators/9-check-login/task.md +++ b/1-js/02-first-steps/11-logical-operators/9-check-login/task.md @@ -6,13 +6,13 @@ importance: 3 Write the code which asks for a login with `prompt`. -If the visitor enters `"Admin"`, then `prompt` for a password, if the input is an empty line or `key:Esc` -- show "Canceled.", if it's another string -- then show "I don't know you". +If the visitor enters `"Admin"`, then `prompt` for a password, if the input is an empty line or `key:Esc` -- show "Canceled", if it's another string -- then show "I don't know you". The password is checked as follows: - If it equals "TheMaster", then show "Welcome!", - Another string -- show "Wrong password", -- For an empty string or cancelled input, show "Canceled." +- For an empty string or cancelled input, show "Canceled" The schema: diff --git a/1-js/02-first-steps/11-logical-operators/article.md b/1-js/02-first-steps/11-logical-operators/article.md index 4932020ae..78c4fd2f1 100644 --- a/1-js/02-first-steps/11-logical-operators/article.md +++ b/1-js/02-first-steps/11-logical-operators/article.md @@ -1,6 +1,6 @@ # Logical operators -There are three logical operators in JavaScript: `||` (OR), `&&` (AND), `!` (NOT). +There are four logical operators in JavaScript: `||` (OR), `&&` (AND), `!` (NOT), `??` (Nullish Coalescing). Here we cover the first three, the `??` operator is in the next article. Although they are called "logical", they can be applied to values of any type, not only boolean. Their result can also be of any type. @@ -64,7 +64,7 @@ if (hour < 10 || hour > 18 || isWeekend) { } ``` -## OR finds the first truthy value +## OR "||" finds the first truthy value [#or-finds-the-first-truthy-value] The logic described above is somewhat classical. Now, let's bring in the "extra" features of JavaScript. @@ -84,16 +84,16 @@ The OR `||` operator does the following: A value is returned in its original form, without the conversion. -In other words, a chain of OR `"||"` returns the first truthy value or the last one if no such value is found. +In other words, a chain of OR `||` returns the first truthy value or the last one if no truthy value is found. For instance: ```js run alert( 1 || 0 ); // 1 (1 is truthy) -alert( true || 'no matter what' ); // (true is truthy) alert( null || 1 ); // 1 (1 is the first truthy value) alert( null || 0 || 1 ); // 1 (the first truthy value) + alert( undefined || null || 0 ); // 0 (all falsy, returns the last value) ``` @@ -101,53 +101,40 @@ This leads to some interesting usage compared to a "pure, classical, boolean-onl 1. **Getting the first truthy value from a list of variables or expressions.** - Imagine we have several variables which can either contain data or be `null/undefined`. How can we find the first one with data? + For instance, we have `firstName`, `lastName` and `nickName` variables, all optional (i.e. can be undefined or have falsy values). - We can use OR `||`: + Let's use OR `||` to choose the one that has the data and show it (or `"Anonymous"` if nothing set): ```js run - let currentUser = null; - let defaultUser = "John"; + let firstName = ""; + let lastName = ""; + let nickName = "SuperCoder"; *!* - let name = currentUser || defaultUser || "unnamed"; + alert( firstName || lastName || nickName || "Anonymous"); // SuperCoder */!* - - alert( name ); // selects "John" – the first truthy value ``` - If both `currentUser` and `defaultUser` were falsy, `"unnamed"` would be the result. -2. **Short-circuit evaluation.** - - Operands can be not only values, but arbitrary expressions. OR evaluates and tests them from left to right. The evaluation stops when a truthy value is reached, and the value is returned. This process is called "a short-circuit evaluation" because it goes as short as possible from left to right. + If all variables were falsy, `"Anonymous"` would show up. - This is clearly seen when the expression given as the second argument has a side effect like a variable assignment. +2. **Short-circuit evaluation.** - In the example below, `x` does not get assigned: + Another feature of OR `||` operator is the so-called "short-circuit" evaluation. - ```js run no-beautify - let x; + It means that `||` processes its arguments until the first truthy value is reached, and then the value is returned immediately, without even touching the other argument. - *!*true*/!* || (x = 1); + The importance of this feature becomes obvious if an operand isn't just a value, but an expression with a side effect, such as a variable assignment or a function call. - alert(x); // undefined, because (x = 1) not evaluated - ``` - - If, instead, the first argument is `false`, `||` evaluates the second one, thus running the assignment: + In the example below, only the second message is printed: ```js run no-beautify - let x; - - *!*false*/!* || (x = 1); - - alert(x); // 1 + *!*true*/!* || alert("not printed"); + *!*false*/!* || alert("printed"); ``` - An assignment is a simple case. Other side effects can also be involved. + In the first line, the OR `||` operator stops the evaluation immediately upon seeing `true`, so the `alert` isn't run. - As we can see, such a use case is a "shorter way of doing `if`". The first operand is converted to boolean. If it's false, the second one is evaluated. - - Most of time, it's better to use a "regular" `if` to keep the code easy to understand, but sometimes this can be handy. + Sometimes, people use this feature to execute commands only if the condition on the left part is falsy. ## && (AND) @@ -186,7 +173,7 @@ if (1 && 0) { // evaluated as true && false ``` -## AND finds the first falsy value +## AND "&&" finds the first falsy value Given multiple AND'ed values: @@ -236,7 +223,8 @@ The precedence of AND `&&` operator is higher than OR `||`. So the code `a && b || c && d` is essentially the same as if the `&&` expressions were in parentheses: `(a && b) || (c && d)`. ```` -Just like OR, the AND `&&` operator can sometimes replace `if`. +````warn header="Don't replace `if` with `||` or `&&`" +Sometimes, people use the AND `&&` operator as a "shorter way to write `if`". For instance: @@ -253,14 +241,12 @@ So we basically have an analogue for: ```js run let x = 1; -if (x > 0) { - alert( 'Greater than zero!' ); -} +if (x > 0) alert( 'Greater than zero!' ); ``` -The variant with `&&` appears shorter. But `if` is more obvious and tends to be a little bit more readable. +Although, the variant with `&&` appears shorter, `if` is more obvious and tends to be a little bit more readable. So we recommend using every construct for its purpose: use `if` if we want `if` and use `&&` if we want AND. +```` -So we recommend using every construct for its purpose: use `if` if we want if and use `&&` if we want AND. ## ! (NOT) diff --git a/1-js/02-first-steps/12-nullish-coalescing-operator/article.md b/1-js/02-first-steps/12-nullish-coalescing-operator/article.md new file mode 100644 index 000000000..0b2f092ab --- /dev/null +++ b/1-js/02-first-steps/12-nullish-coalescing-operator/article.md @@ -0,0 +1,169 @@ +# Nullish coalescing operator '??' + +[recent browser="new"] + +The nullish coalescing operator is written as two question marks `??`. + +As it treats `null` and `undefined` similarly, we'll use a special term here, in this article. For brevity, we'll say that a value is "defined" when it's neither `null` nor `undefined`. + +The result of `a ?? b` is: +- if `a` is defined, then `a`, +- if `a` isn't defined, then `b`. + +In other words, `??` returns the first argument if it's not `null/undefined`. Otherwise, the second one. + +The nullish coalescing operator isn't anything completely new. It's just a nice syntax to get the first "defined" value of the two. + +We can rewrite `result = a ?? b` using the operators that we already know, like this: + +```js +result = (a !== null && a !== undefined) ? a : b; +``` + +Now it should be absolutely clear what `??` does. Let's see where it helps. + +The common use case for `??` is to provide a default value. + +For example, here we show `user` if its value isn't `null/undefined`, otherwise `Anonymous`: + +```js run +let user; + +alert(user ?? "Anonymous"); // Anonymous (user is undefined) +``` + +Here's the example with `user` assigned to a name: + +```js run +let user = "John"; + +alert(user ?? "Anonymous"); // John (user is not null/undefined) +``` + +We can also use a sequence of `??` to select the first value from a list that isn't `null/undefined`. + +Let's say we have a user's data in variables `firstName`, `lastName` or `nickName`. All of them may be not defined, if the user decided not to fill in the corresponding values. + +We'd like to display the user name using one of these variables, or show "Anonymous" if all of them are `null/undefined`. + +Let's use the `??` operator for that: + +```js run +let firstName = null; +let lastName = null; +let nickName = "Supercoder"; + +// shows the first defined value: +*!* +alert(firstName ?? lastName ?? nickName ?? "Anonymous"); // Supercoder +*/!* +``` + +## Comparison with || + +The OR `||` operator can be used in the same way as `??`, as it was described in the [previous chapter](info:logical-operators#or-finds-the-first-truthy-value). + +For example, in the code above we could replace `??` with `||` and still get the same result: + +```js run +let firstName = null; +let lastName = null; +let nickName = "Supercoder"; + +// shows the first truthy value: +*!* +alert(firstName || lastName || nickName || "Anonymous"); // Supercoder +*/!* +``` + +Historically, the OR `||` operator was there first. It's been there since the beginning of JavaScript, so developers were using it for such purposes for a long time. + +On the other hand, the nullish coalescing operator `??` was added to JavaScript only recently, and the reason for that was that people weren't quite happy with `||`. + +The important difference between them is that: +- `||` returns the first *truthy* value. +- `??` returns the first *defined* value. + +In other words, `||` doesn't distinguish between `false`, `0`, an empty string `""` and `null/undefined`. They are all the same -- falsy values. If any of these is the first argument of `||`, then we'll get the second argument as the result. + +In practice though, we may want to use default value only when the variable is `null/undefined`. That is, when the value is really unknown/not set. + +For example, consider this: + +```js run +let height = 0; + +alert(height || 100); // 100 +alert(height ?? 100); // 0 +``` + +- The `height || 100` checks `height` for being a falsy value, and it's `0`, falsy indeed. + - so the result of `||` is the second argument, `100`. +- The `height ?? 100` checks `height` for being `null/undefined`, and it's not, + - so the result is `height` "as is", that is `0`. + +In practice, the zero height is often a valid value, that shouldn't be replaced with the default. So `??` does just the right thing. + +## Precedence + +The precedence of the `??` operator is the same as `||`. They both equal `3` in the [MDN table](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Operator_Precedence#Table). + +That means that, just like `||`, the nullish coalescing operator `??` is evaluated before `=` and `?`, but after most other operations, such as `+`, `*`. + +So we may need to add parentheses in expressions like this: + +```js run +let height = null; +let width = null; + +// important: use parentheses +let area = (height ?? 100) * (width ?? 50); + +alert(area); // 5000 +``` + +Otherwise, if we omit parentheses, then as `*` has the higher precedence than `??`, it would execute first, leading to incorrect results. + +```js +// without parentheses +let area = height ?? 100 * width ?? 50; + +// ...works this way (not what we want): +let area = height ?? (100 * width) ?? 50; +``` + +### Using ?? with && or || + +Due to safety reasons, JavaScript forbids using `??` together with `&&` and `||` operators, unless the precedence is explicitly specified with parentheses. + +The code below triggers a syntax error: + +```js run +let x = 1 && 2 ?? 3; // Syntax error +``` + +The limitation is surely debatable, it was added to the language specification with the purpose to avoid programming mistakes, when people start to switch from `||` to `??`. + +Use explicit parentheses to work around it: + +```js run +*!* +let x = (1 && 2) ?? 3; // Works +*/!* + +alert(x); // 2 +``` + +## Summary + +- The nullish coalescing operator `??` provides a short way to choose the first "defined" value from a list. + + It's used to assign default values to variables: + + ```js + // set height=100, if height is null or undefined + height = height ?? 100; + ``` + +- The operator `??` has a very low precedence, only a bit higher than `?` and `=`, so consider adding parentheses when using it in an expression. +- It's forbidden to use it with `||` or `&&` without explicit parentheses. diff --git a/1-js/02-first-steps/12-while-for/1-loop-last-value/solution.md b/1-js/02-first-steps/13-while-for/1-loop-last-value/solution.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/1-loop-last-value/solution.md rename to 1-js/02-first-steps/13-while-for/1-loop-last-value/solution.md diff --git a/1-js/02-first-steps/12-while-for/1-loop-last-value/task.md b/1-js/02-first-steps/13-while-for/1-loop-last-value/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/1-loop-last-value/task.md rename to 1-js/02-first-steps/13-while-for/1-loop-last-value/task.md diff --git a/1-js/02-first-steps/12-while-for/2-which-value-while/solution.md b/1-js/02-first-steps/13-while-for/2-which-value-while/solution.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/2-which-value-while/solution.md rename to 1-js/02-first-steps/13-while-for/2-which-value-while/solution.md diff --git a/1-js/02-first-steps/12-while-for/2-which-value-while/task.md b/1-js/02-first-steps/13-while-for/2-which-value-while/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/2-which-value-while/task.md rename to 1-js/02-first-steps/13-while-for/2-which-value-while/task.md diff --git a/1-js/02-first-steps/12-while-for/3-which-value-for/solution.md b/1-js/02-first-steps/13-while-for/3-which-value-for/solution.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/3-which-value-for/solution.md rename to 1-js/02-first-steps/13-while-for/3-which-value-for/solution.md diff --git a/1-js/02-first-steps/12-while-for/3-which-value-for/task.md b/1-js/02-first-steps/13-while-for/3-which-value-for/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/3-which-value-for/task.md rename to 1-js/02-first-steps/13-while-for/3-which-value-for/task.md diff --git a/1-js/02-first-steps/12-while-for/4-for-even/solution.md b/1-js/02-first-steps/13-while-for/4-for-even/solution.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/4-for-even/solution.md rename to 1-js/02-first-steps/13-while-for/4-for-even/solution.md diff --git a/1-js/02-first-steps/12-while-for/4-for-even/task.md b/1-js/02-first-steps/13-while-for/4-for-even/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/4-for-even/task.md rename to 1-js/02-first-steps/13-while-for/4-for-even/task.md diff --git a/1-js/02-first-steps/12-while-for/5-replace-for-while/solution.md b/1-js/02-first-steps/13-while-for/5-replace-for-while/solution.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/5-replace-for-while/solution.md rename to 1-js/02-first-steps/13-while-for/5-replace-for-while/solution.md diff --git a/1-js/02-first-steps/12-while-for/5-replace-for-while/task.md b/1-js/02-first-steps/13-while-for/5-replace-for-while/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/5-replace-for-while/task.md rename to 1-js/02-first-steps/13-while-for/5-replace-for-while/task.md diff --git a/1-js/02-first-steps/12-while-for/6-repeat-until-correct/solution.md b/1-js/02-first-steps/13-while-for/6-repeat-until-correct/solution.md similarity index 80% rename from 1-js/02-first-steps/12-while-for/6-repeat-until-correct/solution.md rename to 1-js/02-first-steps/13-while-for/6-repeat-until-correct/solution.md index 2e04a78c4..c7de5f09b 100644 --- a/1-js/02-first-steps/12-while-for/6-repeat-until-correct/solution.md +++ b/1-js/02-first-steps/13-while-for/6-repeat-until-correct/solution.md @@ -10,6 +10,6 @@ do { The loop `do..while` repeats while both checks are truthy: 1. The check for `num <= 100` -- that is, the entered value is still not greater than `100`. -2. The check `&& num` is false when `num` is `null` or a empty string. Then the `while` loop stops too. +2. The check `&& num` is false when `num` is `null` or an empty string. Then the `while` loop stops too. P.S. If `num` is `null` then `num <= 100` is `true`, so without the 2nd check the loop wouldn't stop if the user clicks CANCEL. Both checks are required. diff --git a/1-js/02-first-steps/12-while-for/6-repeat-until-correct/task.md b/1-js/02-first-steps/13-while-for/6-repeat-until-correct/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/6-repeat-until-correct/task.md rename to 1-js/02-first-steps/13-while-for/6-repeat-until-correct/task.md diff --git a/1-js/02-first-steps/12-while-for/7-list-primes/solution.md b/1-js/02-first-steps/13-while-for/7-list-primes/solution.md similarity index 53% rename from 1-js/02-first-steps/12-while-for/7-list-primes/solution.md rename to 1-js/02-first-steps/13-while-for/7-list-primes/solution.md index 9ff0663d7..b4b64b6fa 100644 --- a/1-js/02-first-steps/12-while-for/7-list-primes/solution.md +++ b/1-js/02-first-steps/13-while-for/7-list-primes/solution.md @@ -26,4 +26,4 @@ for (let i = 2; i <= n; i++) { // for each i... } ``` -There's a lot of space to opimize it. For instance, we could look for the divisors from `2` to square root of `i`. But anyway, if we want to be really efficient for large intervals, we need to change the approach and rely on advanced maths and complex algorithms like [Quadratic sieve](https://en.wikipedia.org/wiki/Quadratic_sieve), [General number field sieve](https://en.wikipedia.org/wiki/General_number_field_sieve) etc. +There's a lot of space to optimize it. For instance, we could look for the divisors from `2` to square root of `i`. But anyway, if we want to be really efficient for large intervals, we need to change the approach and rely on advanced maths and complex algorithms like [Quadratic sieve](https://en.wikipedia.org/wiki/Quadratic_sieve), [General number field sieve](https://en.wikipedia.org/wiki/General_number_field_sieve) etc. diff --git a/1-js/02-first-steps/12-while-for/7-list-primes/task.md b/1-js/02-first-steps/13-while-for/7-list-primes/task.md similarity index 100% rename from 1-js/02-first-steps/12-while-for/7-list-primes/task.md rename to 1-js/02-first-steps/13-while-for/7-list-primes/task.md diff --git a/1-js/02-first-steps/12-while-for/article.md b/1-js/02-first-steps/13-while-for/article.md similarity index 83% rename from 1-js/02-first-steps/12-while-for/article.md rename to 1-js/02-first-steps/13-while-for/article.md index 992c21af6..d1b749888 100644 --- a/1-js/02-first-steps/12-while-for/article.md +++ b/1-js/02-first-steps/13-while-for/article.md @@ -6,6 +6,19 @@ For example, outputting goods from a list one after another or just running the *Loops* are a way to repeat the same code multiple times. +```smart header="The for..of and for..in loops" +A small announcement for advanced readers. + +This article covers only basic loops: `while`, `do..while` and `for(..;..;..)`. + +If you came to this article searching for other types of loops, here are the pointers: + +- See [for..in](info:object#forin) to loop over object properties. +- See [for..of](info:array#loops) and [iterables](info:iterable) for looping over arrays and iterable objects. + +Otherwise, please read on. +``` + ## The "while" loop The `while` loop has the following syntax: @@ -17,7 +30,7 @@ while (condition) { } ``` -While the `condition` is `true`, the `code` from the loop body is executed. +While the `condition` is truthy, the `code` from the loop body is executed. For instance, the loop below outputs `i` while `i < 3`: @@ -47,8 +60,8 @@ while (i) { // when i becomes 0, the condition becomes falsy, and the loop stops } ``` -````smart header="Brackets are not required for a single-line body" -If the loop body has a single statement, we can omit the brackets `{…}`: +````smart header="Curly braces are not required for a single-line body" +If the loop body has a single statement, we can omit the curly braces `{…}`: ```js run let i = 3; @@ -84,7 +97,7 @@ This form of syntax should only be used when you want the body of the loop to ex ## The "for" loop -The `for` loop is the most commonly used loop. +The `for` loop is more complex, but it's also the most commonly used loop. It looks like this: @@ -106,13 +119,13 @@ Let's examine the `for` statement part-by-part: | part | | | |-------|----------|----------------------------------------------------------------------------| -| begin | `i = 0` | Executes once upon entering the loop. | +| begin | `let i = 0` | Executes once upon entering the loop. | | condition | `i < 3`| Checked before every loop iteration. If false, the loop stops. | -| step| `i++` | Executes after the body on each iteration but before the condition check. | | body | `alert(i)`| Runs again and again while the condition is truthy. | - +| step| `i++` | Executes after the body on each iteration. | The general loop algorithm works like this: + ``` Run begin → (if condition → run body and run step) @@ -121,6 +134,8 @@ Run begin → ... ``` +That is, `begin` executes once, and then it iterates: after each `condition` test, `body` and `step` are executed. + If you are new to loops, it could help to go back to the example and reproduce how it runs step-by-step on a piece of paper. Here's exactly what happens in our case: @@ -160,10 +175,8 @@ for (i = 0; i < 3; i++) { // use an existing variable alert(i); // 3, visible, because declared outside of the loop ``` - ```` - ### Skipping parts Any part of `for` can be skipped. @@ -210,7 +223,7 @@ But we can force the exit at any time using the special `break` directive. For example, the loop below asks the user for a series of numbers, "breaking" when no number is entered: -```js +```js run let sum = 0; while (true) { @@ -254,7 +267,7 @@ For even values of `i`, the `continue` directive stops executing the body and pa ````smart header="The `continue` directive helps decrease nesting" A loop that shows odd values could look like this: -```js +```js run for (let i = 0; i < 10; i++) { if (i % 2) { @@ -266,7 +279,7 @@ for (let i = 0; i < 10; i++) { From a technical point of view, this is identical to the example above. Surely, we can just wrap the code in an `if` block instead of using `continue`. -But as a side-effect, this created one more level of nesting (the `alert` call inside the curly braces). If the code inside of`if` is longer than a few lines, that may decrease the overall readability. +But as a side effect, this created one more level of nesting (the `alert` call inside the curly braces). If the code inside of `if` is longer than a few lines, that may decrease the overall readability. ```` ````warn header="No `break/continue` to the right side of '?'" @@ -284,13 +297,11 @@ if (i > 5) { ...and rewrite it using a question mark: - ```js no-beautify (i > 5) ? alert(i) : *!*continue*/!*; // continue isn't allowed here ``` -...it stops working. Code like this will give a syntax error: - +...it stops working: there's a syntax error. This is just another reason not to use the question mark operator `?` instead of `if`. ```` @@ -299,7 +310,7 @@ This is just another reason not to use the question mark operator `?` instead of Sometimes we need to break out from multiple nested loops at once. -For example, in the code below we loop over `i` and `j`, prompting for the coordinates `(i, j)` from `(0,0)` to `(3,3)`: +For example, in the code below we loop over `i` and `j`, prompting for the coordinates `(i, j)` from `(0,0)` to `(2,2)`: ```js run no-beautify for (let i = 0; i < 3; i++) { @@ -308,8 +319,7 @@ for (let i = 0; i < 3; i++) { let input = prompt(`Value at coords (${i},${j})`, ''); - // what if I want to exit from here to Done (below)? - + // what if we want to exit from here to Done (below)? } } @@ -318,9 +328,10 @@ alert('Done!'); We need a way to stop the process if the user cancels the input. -The ordinary `break` after `input` would only break the inner loop. That's not sufficient--labels, come to the rescue! +The ordinary `break` after `input` would only break the inner loop. That's not sufficient -- labels, come to the rescue! A *label* is an identifier with a colon before a loop: + ```js labelName: for (...) { ... @@ -342,6 +353,7 @@ The `break <labelName>` statement in the loop below breaks out to the label: // do something with the value... } } + alert('Done!'); ``` @@ -358,17 +370,30 @@ for (let i = 0; i < 3; i++) { ... } The `continue` directive can also be used with a label. In this case, code execution jumps to the next iteration of the labeled loop. -````warn header="Labels are not a \"goto\"" +````warn header="Labels do not allow to \"jump\" anywhere" Labels do not allow us to jump into an arbitrary place in the code. For example, it is impossible to do this: + ```js -break label; // jumps to label? No. +break label; // jump to the label below (doesn't work) label: for (...) ``` -A call to `break/continue` is only possible from inside a loop and the label must be somewhere above the directive. +A `break` directive must be inside a code block. Technically, any labelled code block will do, e.g.: + +```js +label: { + // ... + break label; // works + // ... +} +``` + +...Although, 99.9% of the time `break` is used inside loops, as we've seen in the examples above. + +A `continue` is only possible from inside a loop. ```` ## Summary diff --git a/1-js/02-first-steps/14-function-basics/1-if-else-required/solution.md b/1-js/02-first-steps/14-function-basics/1-if-else-required/solution.md deleted file mode 100644 index e41c80418..000000000 --- a/1-js/02-first-steps/14-function-basics/1-if-else-required/solution.md +++ /dev/null @@ -1 +0,0 @@ -No difference. \ No newline at end of file diff --git a/1-js/02-first-steps/14-function-basics/function_basics.png b/1-js/02-first-steps/14-function-basics/function_basics.png deleted file mode 100644 index f5e6f9418..000000000 Binary files a/1-js/02-first-steps/14-function-basics/function_basics.png and /dev/null differ diff --git a/1-js/02-first-steps/14-function-basics/function_basics@2x.png b/1-js/02-first-steps/14-function-basics/function_basics@2x.png deleted file mode 100644 index c31b2636a..000000000 Binary files a/1-js/02-first-steps/14-function-basics/function_basics@2x.png and /dev/null differ diff --git a/1-js/02-first-steps/13-switch/1-rewrite-switch-if-else/solution.md b/1-js/02-first-steps/14-switch/1-rewrite-switch-if-else/solution.md similarity index 100% rename from 1-js/02-first-steps/13-switch/1-rewrite-switch-if-else/solution.md rename to 1-js/02-first-steps/14-switch/1-rewrite-switch-if-else/solution.md diff --git a/1-js/02-first-steps/13-switch/1-rewrite-switch-if-else/task.md b/1-js/02-first-steps/14-switch/1-rewrite-switch-if-else/task.md similarity index 100% rename from 1-js/02-first-steps/13-switch/1-rewrite-switch-if-else/task.md rename to 1-js/02-first-steps/14-switch/1-rewrite-switch-if-else/task.md diff --git a/1-js/02-first-steps/13-switch/2-rewrite-if-switch/solution.md b/1-js/02-first-steps/14-switch/2-rewrite-if-switch/solution.md similarity index 100% rename from 1-js/02-first-steps/13-switch/2-rewrite-if-switch/solution.md rename to 1-js/02-first-steps/14-switch/2-rewrite-if-switch/solution.md diff --git a/1-js/02-first-steps/13-switch/2-rewrite-if-switch/task.md b/1-js/02-first-steps/14-switch/2-rewrite-if-switch/task.md similarity index 100% rename from 1-js/02-first-steps/13-switch/2-rewrite-if-switch/task.md rename to 1-js/02-first-steps/14-switch/2-rewrite-if-switch/task.md diff --git a/1-js/02-first-steps/13-switch/article.md b/1-js/02-first-steps/14-switch/article.md similarity index 95% rename from 1-js/02-first-steps/13-switch/article.md rename to 1-js/02-first-steps/14-switch/article.md index 258f24068..d86babcec 100644 --- a/1-js/02-first-steps/13-switch/article.md +++ b/1-js/02-first-steps/14-switch/article.md @@ -47,7 +47,7 @@ switch (a) { break; */!* case 5: - alert( 'Too large' ); + alert( 'Too big' ); break; default: alert( "I don't know such values" ); @@ -117,7 +117,7 @@ Several variants of `case` which share the same code can be grouped. For example, if we want the same code to run for `case 3` and `case 5`: ```js run no-beautify -let a = 2 + 2; +let a = 3; switch (a) { case 4: @@ -125,7 +125,7 @@ switch (a) { break; *!* - case 3: // (*) grouped two cases + case 3: // (*) grouped two cases case 5: alert('Wrong!'); alert("Why don't you take a math class?"); @@ -139,7 +139,7 @@ switch (a) { Now both `3` and `5` show the same message. -The ability to "group" cases is a side-effect of how `switch/case` works without `break`. Here the execution of `case 3` starts from the line `(*)` and goes through `case 5`, because there's no `break`. +The ability to "group" cases is a side effect of how `switch/case` works without `break`. Here the execution of `case 3` starts from the line `(*)` and goes through `case 5`, because there's no `break`. ## Type matters diff --git a/1-js/02-first-steps/15-function-basics/1-if-else-required/solution.md b/1-js/02-first-steps/15-function-basics/1-if-else-required/solution.md new file mode 100644 index 000000000..e3a0df77c --- /dev/null +++ b/1-js/02-first-steps/15-function-basics/1-if-else-required/solution.md @@ -0,0 +1,3 @@ +No difference! + +In both cases, `return confirm('Did parents allow you?')` executes exactly when the `if` condition is falsy. \ No newline at end of file diff --git a/1-js/02-first-steps/14-function-basics/1-if-else-required/task.md b/1-js/02-first-steps/15-function-basics/1-if-else-required/task.md similarity index 100% rename from 1-js/02-first-steps/14-function-basics/1-if-else-required/task.md rename to 1-js/02-first-steps/15-function-basics/1-if-else-required/task.md diff --git a/1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/solution.md b/1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/solution.md similarity index 89% rename from 1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/solution.md rename to 1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/solution.md index c8ee9618f..e48502642 100644 --- a/1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/solution.md +++ b/1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/solution.md @@ -14,4 +14,4 @@ function checkAge(age) { } ``` -Note that the parentheses around `age > 18` are not required here. They exist for better readabilty. +Note that the parentheses around `age > 18` are not required here. They exist for better readability. diff --git a/1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/task.md b/1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/task.md similarity index 85% rename from 1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/task.md rename to 1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/task.md index 523bb127a..46da079c0 100644 --- a/1-js/02-first-steps/14-function-basics/2-rewrite-function-question-or/task.md +++ b/1-js/02-first-steps/15-function-basics/2-rewrite-function-question-or/task.md @@ -13,7 +13,7 @@ function checkAge(age) { if (age > 18) { return true; } else { - return confirm('Do you have your parents permission to access this page?'); + return confirm('Did parents allow you?'); } } ``` diff --git a/1-js/02-first-steps/14-function-basics/3-min/solution.md b/1-js/02-first-steps/15-function-basics/3-min/solution.md similarity index 100% rename from 1-js/02-first-steps/14-function-basics/3-min/solution.md rename to 1-js/02-first-steps/15-function-basics/3-min/solution.md diff --git a/1-js/02-first-steps/14-function-basics/3-min/task.md b/1-js/02-first-steps/15-function-basics/3-min/task.md similarity index 100% rename from 1-js/02-first-steps/14-function-basics/3-min/task.md rename to 1-js/02-first-steps/15-function-basics/3-min/task.md diff --git a/1-js/02-first-steps/14-function-basics/4-pow/solution.md b/1-js/02-first-steps/15-function-basics/4-pow/solution.md similarity index 75% rename from 1-js/02-first-steps/14-function-basics/4-pow/solution.md rename to 1-js/02-first-steps/15-function-basics/4-pow/solution.md index 5ef20c386..19fe9011f 100644 --- a/1-js/02-first-steps/14-function-basics/4-pow/solution.md +++ b/1-js/02-first-steps/15-function-basics/4-pow/solution.md @@ -14,10 +14,8 @@ let x = prompt("x?", ''); let n = prompt("n?", ''); if (n < 1) { - alert(`Power ${n} is not supported, - use an integer greater than 0`); + alert(`Power ${n} is not supported, use a positive integer`); } else { alert( pow(x, n) ); } ``` - diff --git a/1-js/02-first-steps/14-function-basics/4-pow/task.md b/1-js/02-first-steps/15-function-basics/4-pow/task.md similarity index 100% rename from 1-js/02-first-steps/14-function-basics/4-pow/task.md rename to 1-js/02-first-steps/15-function-basics/4-pow/task.md diff --git a/1-js/02-first-steps/14-function-basics/article.md b/1-js/02-first-steps/15-function-basics/article.md similarity index 71% rename from 1-js/02-first-steps/14-function-basics/article.md rename to 1-js/02-first-steps/15-function-basics/article.md index 18833cbf1..415fed3e0 100644 --- a/1-js/02-first-steps/14-function-basics/article.md +++ b/1-js/02-first-steps/15-function-basics/article.md @@ -20,9 +20,13 @@ function showMessage() { } ``` -The `function` keyword goes first, then goes the *name of the function*, then a list of *parameters* between the parentheses (empty in the example above) and finally the code of the function, also named "the function body", between curly braces. +The `function` keyword goes first, then goes the *name of the function*, then a list of *parameters* between the parentheses (comma-separated, empty in the example above, we'll see examples later) and finally the code of the function, also named "the function body", between curly braces. - +```js +function name(parameter1, parameter2, ... parameterN) { + // body +} +``` Our new function can be called by its name: `showMessage()`. @@ -101,7 +105,7 @@ showMessage(); alert( userName ); // *!*Bob*/!*, the value was modified by the function ``` -The outer variable is only used if there's no local one. So an occasional modification may happen if we forget `let`. +The outer variable is only used if there's no local one. If a same-named variable is declared inside the function then it *shadows* the outer one. For instance, in the code below the function uses the local `userName`. The outer one is ignored: @@ -128,31 +132,28 @@ Variables declared outside of any function, such as the outer `userName` in the Global variables are visible from any function (unless shadowed by locals). -Usually, a function declares all variables specific to its task. Global variables only store project-level data, and it's important that these variables are accessible from anywhere. Modern code has few or no globals. Most variables reside in their functions. +It's a good practice to minimize the use of global variables. Modern code has few or no globals. Most variables reside in their functions. Sometimes though, they can be useful to store project-level data. ``` ## Parameters -We can pass arbitrary data to functions using parameters (also called *function arguments*) . +We can pass arbitrary data to functions using parameters. In the example below, the function has two parameters: `from` and `text`. ```js run -function showMessage(*!*from, text*/!*) { // arguments: from, text +function showMessage(*!*from, text*/!*) { // parameters: from, text alert(from + ': ' + text); } -*!* -showMessage('Ann', 'Hello!'); // Ann: Hello! (*) -showMessage('Ann', "What's up?"); // Ann: What's up? (**) -*/!* +*!*showMessage('Ann', 'Hello!');*/!* // Ann: Hello! (*) +*!*showMessage('Ann', "What's up?");*/!* // Ann: What's up? (**) ``` When the function is called in lines `(*)` and `(**)`, the given values are copied to local variables `from` and `text`. Then the function uses them. Here's one more example: we have a variable `from` and pass it to the function. Please note: the function changes `from`, but the change is not seen outside, because a function always gets a copy of the value: - ```js run function showMessage(from, text) { @@ -171,9 +172,21 @@ showMessage(from, "Hello"); // *Ann*: Hello alert( from ); // Ann ``` +When a value is passed as a function parameter, it's also called an *argument*. + +In other words, to put these terms straight: + +- A parameter is the variable listed inside the parentheses in the function declaration (it's a declaration time term). +- An argument is the value that is passed to the function when it is called (it's a call time term). + +We declare functions listing their parameters, then call them passing arguments. + +In the example above, one might say: "the function `showMessage` is declared with two parameters, then called with two arguments: `from` and `"Hello"`". + + ## Default values -If a parameter is not provided, then its value becomes `undefined`. +If a function is called, but an argument is not provided, then the corresponding value becomes `undefined`. For instance, the aforementioned function `showMessage(from, text)` can be called with a single argument: @@ -181,9 +194,9 @@ For instance, the aforementioned function `showMessage(from, text)` can be calle showMessage("Ann"); ``` -That's not an error. Such a call would output `"Ann: undefined"`. There's no `text`, so it's assumed that `text === undefined`. +That's not an error. Such a call would output `"*Ann*: undefined"`. As the value for `text` isn't passed, it becomes `undefined`. -If we want to use a "default" `text` in this case, then we can specify it after `=`: +We can specify the so-called "default" (to use if omitted) value for a parameter in the function declaration, using `=`: ```js run function showMessage(from, *!*text = "no text given"*/!*) { @@ -193,7 +206,13 @@ function showMessage(from, *!*text = "no text given"*/!*) { showMessage("Ann"); // Ann: no text given ``` -Now if the `text` parameter is not passed, it will get the value `"no text given"` +Now if the `text` parameter is not passed, it will get the value `"no text given"`. + +The default value also jumps in if the parameter exists, but strictly equals `undefined`, like this: + +```js +showMessage("Ann", undefined); // Ann: no text given +``` Here `"no text given"` is a string, but it can be a more complex expression, which is only evaluated and assigned if the parameter is missing. So, this is also possible: @@ -205,16 +224,19 @@ function showMessage(from, text = anotherFunction()) { ``` ```smart header="Evaluation of default parameters" +In JavaScript, a default parameter is evaluated every time the function is called without the respective parameter. -In JavaScript, a default parameter is evaluated every time the function is called without the respective parameter. In the example above, `anotherFunction()` is called every time `showMessage()` is called without the `text` parameter. This is in contrast to some other languages like Python, where any default parameters are evaluated only once during the initial interpretation. +In the example above, `anotherFunction()` isn't called at all, if the `text` parameter is provided. +On the other hand, it's independently called every time when `text` is missing. ``` +````smart header="Default parameters in old JavaScript code" +Several years ago, JavaScript didn't support the syntax for default parameters. So people used other ways to specify them. -````smart header="Default parameters old-style" -Old editions of JavaScript did not support default parameters. So there are alternative ways to support them, that you can find mostly in the old scripts. +Nowadays, we can come across them in old scripts. -For instance, an explicit check for being `undefined`: +For example, an explicit check for `undefined`: ```js function showMessage(from, text) { @@ -228,19 +250,63 @@ function showMessage(from, text) { } ``` -...Or the `||` operator: +...Or using the `||` operator: ```js function showMessage(from, text) { - // if text is falsy then text gets the "default" value + // If the value of text is falsy, assign the default value + // this assumes that text == "" is the same as no text at all text = text || 'no text given'; ... } ``` +```` -```` +### Alternative default parameters +Sometimes it makes sense to assign default values for parameters at a later stage after the function declaration. + +We can check if the parameter is passed during the function execution, by comparing it with `undefined`: + +```js run +function showMessage(text) { + // ... + +*!* + if (text === undefined) { // if the parameter is missing + text = 'empty message'; + } +*/!* + + alert(text); +} + +showMessage(); // empty message +``` + +...Or we could use the `||` operator: + +```js +function showMessage(text) { + // if text is undefined or otherwise falsy, set it to 'empty' + text = text || 'empty'; + ... +} +``` + +Modern JavaScript engines support the [nullish coalescing operator](info:nullish-coalescing-operator) `??`, it's better when most falsy values, such as `0`, should be considered "normal": + +```js run +function showCount(count) { + // if count is undefined or null, show "unknown" + alert(count ?? "unknown"); +} + +showCount(0); // 0 +showCount(null); // unknown +showCount(); // unknown +``` ## Returning a value @@ -263,7 +329,7 @@ There may be many occurrences of `return` in a single function. For instance: ```js run function checkAge(age) { - if (age > 18) { + if (age >= 18) { *!* return true; */!* @@ -335,7 +401,19 @@ That doesn't work, because JavaScript assumes a semicolon after `return`. That'l return*!*;*/!* (some + long + expression + or + whatever * f(a) + f(b)) ``` -So, it effectively becomes an empty return. We should put the value on the same line instead. + +So, it effectively becomes an empty return. + +If we want the returned expression to wrap across multiple lines, we should start it at the same line as `return`. Or at least put the opening parentheses there as follows: + +```js +return ( + some + long + expression + + or + + whatever * f(a) + f(b) + ) +``` +And it will work just as we expect it to. ```` ## Naming a function [#function-naming] @@ -376,15 +454,15 @@ A few examples of breaking this rule: - `createForm` -- would be bad if it modifies the document, adding a form to it (should only create it and return). - `checkPermission` -- would be bad if it displays the `access granted/denied` message (should only perform the check and return the result). -These examples assume common meanings of prefixes. What they mean for you is determined by you and your team. Maybe it's pretty normal for your code to behave differently. But you should have a firm understanding of what a prefix means, what a prefixed function can and cannot do. All same-prefixed functions should obey the rules. And the team should share the knowledge. +These examples assume common meanings of prefixes. You and your team are free to agree on other meanings, but usually they're not much different. In any case, you should have a firm understanding of what a prefix means, what a prefixed function can and cannot do. All same-prefixed functions should obey the rules. And the team should share the knowledge. ``` ```smart header="Ultrashort function names" Functions that are used *very often* sometimes have ultrashort names. -For example, the [jQuery](http://jquery.com) framework defines a function with `$`. The [Lodash](http://lodash.com/) library has its core function named `_`. +For example, the [jQuery](https://jquery.com/) framework defines a function with `$`. The [Lodash](https://lodash.com/) library has its core function named `_`. -These are exceptions. Generally functions names should be concise and descriptive. +These are exceptions. Generally function names should be concise and descriptive. ``` ## Functions == Comments @@ -450,7 +528,7 @@ function name(parameters, delimited, by, comma) { To make the code clean and easy to understand, it's recommended to use mainly local variables and parameters in the function, not outer variables. -It is always easier to understand a function which gets parameters, works with them and returns a result than a function which gets no parameters, but modifies outer variables as a side-effect. +It is always easier to understand a function which gets parameters, works with them and returns a result than a function which gets no parameters, but modifies outer variables as a side effect. Function naming: diff --git a/1-js/02-first-steps/15-function-expressions-arrows/article.md b/1-js/02-first-steps/16-function-expressions/article.md similarity index 54% rename from 1-js/02-first-steps/15-function-expressions-arrows/article.md rename to 1-js/02-first-steps/16-function-expressions/article.md index b4ea19bac..c6dd891bd 100644 --- a/1-js/02-first-steps/15-function-expressions-arrows/article.md +++ b/1-js/02-first-steps/16-function-expressions/article.md @@ -1,4 +1,4 @@ -# Function expressions and arrows +# Function expressions In JavaScript, a function is not a "magical language structure", but a special kind of value. @@ -12,7 +12,9 @@ function sayHi() { There is another syntax for creating a function that is called a *Function Expression*. -It looks like this: +It allows us to create a new function in the middle of any expression. + +For example: ```js let sayHi = function() { @@ -20,10 +22,19 @@ let sayHi = function() { }; ``` -Here, the function is created and assigned to the variable explicitly, like any other value. No matter how the function is defined, it's just a value stored in the variable `sayHi`. +Here we can see a variable `sayHi` getting a value, the new function, created as `function() { alert("Hello"); }`. + +As the function creation happens in the context of the assignment expression (to the right side of `=`), this is a *Function Expression*. + +Please note, there's no name after the `function` keyword. Omitting a name is allowed for Function Expressions. + +Here we immediately assign it to the variable, so the meaning of these code samples is the same: "create a function and put it into the variable `sayHi`". + +In more advanced situations, that we'll come across later, a function may be created and immediately called or scheduled for a later execution, not stored anywhere, thus remaining anonymous. +## Function is a value -The meaning of these code samples is the same: "create a function and put it into the variable `sayHi`". +Let's reiterate: no matter how the function is created, a function is a value. Both examples above store a function in the `sayHi` variable. We can even print out that value using `alert`: @@ -41,7 +52,7 @@ Please note that the last line does not run the function, because there are no p In JavaScript, a function is a value, so we can deal with it as a value. The code above shows its string representation, which is the source code. -It is a special value of course, in the sense that we can call it like `sayHi()`. +Surely, a function is a special value, in the sense that we can call it like `sayHi()`. But it's still a value. So we can work with it like with other kinds of values. @@ -61,25 +72,25 @@ sayHi(); // Hello // this still works too (why wouldn't it) Here's what happens above in detail: 1. The Function Declaration `(1)` creates the function and puts it into the variable named `sayHi`. -2. Line `(2)` copies it into the variable `func`. - - Please note again: there are no parentheses after `sayHi`. If there were, then `func = sayHi()` would write *the result of the call* `sayHi()` into `func`, not *the function* `sayHi` itself. +2. Line `(2)` copies it into the variable `func`. Please note again: there are no parentheses after `sayHi`. If there were, then `func = sayHi()` would write *the result of the call* `sayHi()` into `func`, not *the function* `sayHi` itself. 3. Now the function can be called as both `sayHi()` and `func()`. -Note that we could also have used a Function Expression to declare `sayHi`, in the first line: +We could also have used a Function Expression to declare `sayHi`, in the first line: ```js -let sayHi = function() { ... }; +let sayHi = function() { // (1) create + alert( "Hello" ); +}; -let func = sayHi; +let func = sayHi; //(2) // ... ``` -Everything would work the same. Even more obvious what's going on, right? +Everything would work the same. ````smart header="Why is there a semicolon at the end?" -You might wonder, why does Function Expression have a semicolon `;` at the end, but Function Declaration does not: +You might wonder, why do Function Expressions have a semicolon `;` at the end, but Function Declarations do not: ```js function sayHi() { @@ -91,9 +102,9 @@ let sayHi = function() { }*!*;*/!* ``` -The answer is simple: -- There's no need for `;` at the end of code blocks and syntax structures that use them like `if { ... }`, `for { }`, `function f { }` etc. -- A Function Expression is used inside the statement: `let sayHi = ...;`, as a value. It's not a code block. The semicolon `;` is recommended at the end of statements, no matter what is the value. So the semicolon here is not related to the Function Expression itself in any way, it just terminates the statement. +The answer is simple: a Function Expression is created here as `function(…) {…}` inside the assignment statement: `let sayHi = …;`. The semicolon `;` is recommended at the end of the statement, it's not a part of the function syntax. + +The semicolon would be there for a simpler assignment, such as `let sayHi = 5;`, and it's also there for a function assignment. ```` ## Callback functions @@ -133,13 +144,13 @@ function showCancel() { ask("Do you agree?", showOk, showCancel); ``` -Before we explore how we can write it in a much shorter way, let's note that in the browser (and on the server-side in some cases) such functions are quite popular. The major difference between a real-life implementation and the example above is that real-life functions use more complex ways to interact with the user than a simple `confirm`. In the browser, such a function usually draws a nice-looking question window. But that's another story. +In practice, such functions are quite useful. The major difference between a real-life `ask` and the example above is that real-life functions use more complex ways to interact with the user than a simple `confirm`. In the browser, such functions usually draw a nice-looking question window. But that's another story. -**The arguments of `ask` are called *callback functions* or just *callbacks*.** +**The arguments `showOk` and `showCancel` of `ask` are called *callback functions* or just *callbacks*.** -The idea is that we pass a function and expect it to be "called back" later if necessary. In our case, `showOk` becomes the callback for the "yes" answer, and `showCancel` for the "no" answer. +The idea is that we pass a function and expect it to be "called back" later if necessary. In our case, `showOk` becomes the callback for "yes" answer, and `showCancel` for "no" answer. -We can use Function Expressions to write the same function much shorter: +We can use Function Expressions to write an equivalent, shorter function: ```js run no-beautify function ask(question, yes, no) { @@ -156,12 +167,10 @@ ask( */!* ``` - Here, functions are declared right inside the `ask(...)` call. They have no name, and so are called *anonymous*. Such functions are not accessible outside of `ask` (because they are not assigned to variables), but that's just what we want here. Such code appears in our scripts very naturally, it's in the spirit of JavaScript. - ```smart header="A function is a value representing an \"action\"" Regular values like strings or numbers represent the *data*. @@ -175,9 +184,9 @@ We can pass it between variables and run when we want. Let's formulate the key differences between Function Declarations and Expressions. -First, the syntax: how to see what is what in the code. +First, the syntax: how to differentiate between them in the code. -- *Function Declaration:* a function, declared as a separate statement, in the main code flow. +- *Function Declaration:* a function, declared as a separate statement, in the main code flow: ```js // Function Declaration @@ -185,8 +194,8 @@ First, the syntax: how to see what is what in the code. return a + b; } ``` -- *Function Expression:* a function, created inside an expression or inside another syntax construct. Here, the function is created at the right side of the "assignment expression" `=`: - +- *Function Expression:* a function, created inside an expression or inside another syntax construct. Here, the function is created on the right side of the "assignment expression" `=`: + ```js // Function Expression let sum = function(a, b) { @@ -196,19 +205,19 @@ First, the syntax: how to see what is what in the code. The more subtle difference is *when* a function is created by the JavaScript engine. -**A Function Expression is created when the execution reaches it and is usable from then on.** +**A Function Expression is created when the execution reaches it and is usable only from that moment.** Once the execution flow passes to the right side of the assignment `let sum = function…` -- here we go, the function is created and can be used (assigned, called, etc. ) from now on. Function Declarations are different. -**A Function Declaration is usable in the whole script/code block.** +**A Function Declaration can be called earlier than it is defined.** -In other words, when JavaScript *prepares* to run the script or a code block, it first looks for Function Declarations in it and creates the functions. We can think of it as an "initialization stage". +For example, a global Function Declaration is visible in the whole script, no matter where it is. -And after all of the Function Declarations are processed, the execution goes on. +That's due to internal algorithms. When JavaScript prepares to run the script, it first looks for global Function Declarations in it and creates the functions. We can think of it as an "initialization stage". -As a result, a function declared as a Function Declaration can be called earlier than it is defined. +And after all Function Declarations are processed, the code is executed. So it has access to these functions. For example, this works: @@ -224,7 +233,7 @@ function sayHi(name) { The Function Declaration `sayHi` is created when JavaScript is preparing to start the script and is visible everywhere in it. -...If it was a Function Expression, then it wouldn't work: +...If it were a Function Expression, then it wouldn't work: ```js run refresh untrusted *!* @@ -238,13 +247,13 @@ let sayHi = function(name) { // (*) no magic any more Function Expressions are created when the execution reaches them. That would happen only in the line `(*)`. Too late. -**When a Function Declaration is made within a code block, it is visible everywhere inside that block. But not outside of it.** +Another special feature of Function Declarations is their block scope. -Sometimes that's handy to declare a local function only needed in that block alone. But that feature may also cause problems. +**In strict mode, when a Function Declaration is within a code block, it's visible everywhere inside that block. But not outside of it.** For instance, let's imagine that we need to declare a function `welcome()` depending on the `age` variable that we get during runtime. And then we plan to use it some time later. -The code below doesn't work: +If we use Function Declaration, it won't work as intended: ```js run let age = prompt("What is your age?", 18); @@ -282,7 +291,7 @@ if (age < 18) { welcome(); // \ (runs) */!* // | - function welcome() { // | + function welcome() { // | alert("Hello!"); // | Function Declaration is available } // | everywhere in the block where it's declared // | @@ -292,7 +301,7 @@ if (age < 18) { } else { - function welcome() { // for age = 16, this "welcome" is never created + function welcome() { alert("Greetings!"); } } @@ -309,7 +318,7 @@ What can we do to make `welcome` visible outside of `if`? The correct approach would be to use a Function Expression and assign `welcome` to the variable that is declared outside of `if` and has the proper visibility. -Now it works as intended: +This code works as intended: ```js run let age = prompt("What is your age?", 18); @@ -350,113 +359,12 @@ welcome(); // ok now ``` -```smart header="When should you choose Function Declaration versus Function Expression?" -As a rule of thumb, when we need to declare a function, the first to consider is Function Declaration syntax, the one we used before. It gives more freedom in how to organize our code, because we can call such functions before they are declared. - -It's also a little bit easier to look up `function f(…) {…}` in the code than `let f = function(…) {…}`. Function Declarations are more "eye-catching". - -...But if a Function Declaration does not suit us for some reason (we've seen an example above), then Function Expression should be used. -``` - - -## Arrow functions [#arrow-functions] +```smart header="When to choose Function Declaration versus Function Expression?" +As a rule of thumb, when we need to declare a function, the first thing to consider is Function Declaration syntax. It gives more freedom in how to organize our code, because we can call such functions before they are declared. -There's one more very simple and concise syntax for creating functions, that's often better than Function Expressions. It's called "arrow functions", because it looks like this: +That's also better for readability, as it's easier to look up `function f(…) {…}` in the code than `let f = function(…) {…};`. Function Declarations are more "eye-catching". - -```js -let func = (arg1, arg2, ...argN) => expression -``` - -...This creates a function `func` that has arguments `arg1..argN`, evaluates the `expression` on the right side with their use and returns its result. - -In other words, it's roughly the same as: - -```js -let func = function(arg1, arg2, ...argN) { - return expression; -}; -``` - -...But much more concise. - -Let's see an example: - -```js run -let sum = (a, b) => a + b; - -/* The arrow function is a shorter form of: - -let sum = function(a, b) { - return a + b; -}; -*/ - -alert( sum(1, 2) ); // 3 - -``` - -If we have only one argument, then parentheses can be omitted, making that even shorter: - -```js run -// same as -// let double = function(n) { return n * 2 } -*!* -let double = n => n * 2; -*/!* - -alert( double(3) ); // 6 -``` - -If there are no arguments, parentheses should be empty (but they should be present): - -```js run -let sayHi = () => alert("Hello!"); - -sayHi(); -``` - -Arrow functions can be used in the same way as Function Expressions. - -For instance, here's the rewritten example with `welcome()`: - -```js run -let age = prompt("What is your age?", 18); - -let welcome = (age < 18) ? - () => alert('Hello') : - () => alert("Greetings!"); - -welcome(); // ok now -``` - -Arrow functions may appear unfamiliar and not very readable at first, but that quickly changes as the eyes get used to the structure. - -They are very convenient for simple one-line actions, when we're just too lazy to write many words. - -```smart header="Multiline arrow functions" - -The examples above took arguments from the left of `=>` and evaluated the right-side expression with them. - -Sometimes we need something a little bit more complex, like multiple expressions or statements. It is also possible, but we should enclose them in curly braces. Then use a normal `return` within them. - -Like this: - -```js run -let sum = (a, b) => { // the curly brace opens a multiline function - let result = a + b; -*!* - return result; // if we use curly braces, use return to get results -*/!* -}; - -alert( sum(1, 2) ); // 3 -``` - -```smart header="More to come" -Here we praised arrow functions for brevity. But that's not all! Arrow functions have other interesting features. We'll return to them later in the chapter <info:arrow-functions>. - -For now, we can already use them for one-line actions and callbacks. +...But if a Function Declaration does not suit us for some reason, or we need a conditional declaration (we've just seen an example), then Function Expression should be used. ``` ## Summary @@ -467,12 +375,6 @@ For now, we can already use them for one-line actions and callbacks. - Function Declarations are processed before the code block is executed. They are visible everywhere in the block. - Function Expressions are created when the execution flow reaches them. - In most cases when we need to declare a function, a Function Declaration is preferable, because it is visible prior to the declaration itself. That gives us more flexibility in code organization, and is usually more readable. So we should use a Function Expression only when a Function Declaration is not fit for the task. We've seen a couple of examples of that in this chapter, and will see more in the future. - -Arrow functions are handy for one-liners. They come in two flavors: - -1. Without curly braces: `(...args) => expression` -- the right side is an expression: the function evaluates it and returns the result. -2. With curly braces: `(...args) => { body }` -- brackets allow us to write multiple statements inside the function, but we need an explicit `return` to return something. diff --git a/1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/solution.md b/1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/solution.md similarity index 86% rename from 1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/solution.md rename to 1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/solution.md index 3ea112473..041db18bc 100644 --- a/1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/solution.md +++ b/1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/solution.md @@ -1,7 +1,7 @@ ```js run function ask(question, yes, no) { - if (confirm(question)) yes() + if (confirm(question)) yes(); else no(); } diff --git a/1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/task.md b/1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/task.md similarity index 68% rename from 1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/task.md rename to 1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/task.md index a888ac157..e18c08a83 100644 --- a/1-js/02-first-steps/15-function-expressions-arrows/1-rewrite-arrow/task.md +++ b/1-js/02-first-steps/17-arrow-functions-basics/1-rewrite-arrow/task.md @@ -1,11 +1,11 @@ # Rewrite with arrow functions -Replace Function Expressions with arrow functions in the code: +Replace Function Expressions with arrow functions in the code below: ```js run function ask(question, yes, no) { - if (confirm(question)) yes() + if (confirm(question)) yes(); else no(); } diff --git a/1-js/02-first-steps/17-arrow-functions-basics/article.md b/1-js/02-first-steps/17-arrow-functions-basics/article.md new file mode 100644 index 000000000..50c0d475d --- /dev/null +++ b/1-js/02-first-steps/17-arrow-functions-basics/article.md @@ -0,0 +1,111 @@ +# Arrow functions, the basics + +There's another very simple and concise syntax for creating functions, that's often better than Function Expressions. + +It's called "arrow functions", because it looks like this: + +```js +let func = (arg1, arg2, ..., argN) => expression; +``` + +This creates a function `func` that accepts arguments `arg1..argN`, then evaluates the `expression` on the right side with their use and returns its result. + +In other words, it's the shorter version of: + +```js +let func = function(arg1, arg2, ..., argN) { + return expression; +}; +``` + +Let's see a concrete example: + +```js run +let sum = (a, b) => a + b; + +/* This arrow function is a shorter form of: + +let sum = function(a, b) { + return a + b; +}; +*/ + +alert( sum(1, 2) ); // 3 +``` + +As you can see, `(a, b) => a + b` means a function that accepts two arguments named `a` and `b`. Upon the execution, it evaluates the expression `a + b` and returns the result. + +- If we have only one argument, then parentheses around parameters can be omitted, making that even shorter. + + For example: + + ```js run + *!* + let double = n => n * 2; + // roughly the same as: let double = function(n) { return n * 2 } + */!* + + alert( double(3) ); // 6 + ``` + +- If there are no arguments, parentheses are empty, but they must be present: + + ```js run + let sayHi = () => alert("Hello!"); + + sayHi(); + ``` + +Arrow functions can be used in the same way as Function Expressions. + +For instance, to dynamically create a function: + +```js run +let age = prompt("What is your age?", 18); + +let welcome = (age < 18) ? + () => alert('Hello!') : + () => alert("Greetings!"); + +welcome(); +``` + +Arrow functions may appear unfamiliar and not very readable at first, but that quickly changes as the eyes get used to the structure. + +They are very convenient for simple one-line actions, when we're just too lazy to write many words. + +## Multiline arrow functions + +The arrow functions that we've seen so far were very simple. They took arguments from the left of `=>`, evaluated and returned the right-side expression with them. + +Sometimes we need a more complex function, with multiple expressions and statements. In that case, we can enclose them in curly braces. The major difference is that curly braces require a `return` within them to return a value (just like a regular function does). + +Like this: + +```js run +let sum = (a, b) => { // the curly brace opens a multiline function + let result = a + b; +*!* + return result; // if we use curly braces, then we need an explicit "return" +*/!* +}; + +alert( sum(1, 2) ); // 3 +``` + +```smart header="More to come" +Here we praised arrow functions for brevity. But that's not all! + +Arrow functions have other interesting features. + +To study them in-depth, we first need to get to know some other aspects of JavaScript, so we'll return to arrow functions later in the chapter <info:arrow-functions>. + +For now, we can already use arrow functions for one-line actions and callbacks. +``` + +## Summary + +Arrow functions are handy for simple actions, especially for one-liners. They come in two flavors: + +1. Without curly braces: `(...args) => expression` -- the right side is an expression: the function evaluates it and returns the result. Parentheses can be omitted, if there's only a single argument, e.g. `n => n*2`. +2. With curly braces: `(...args) => { body }` -- brackets allow us to write multiple statements inside the function, but we need an explicit `return` to return something. diff --git a/1-js/02-first-steps/16-javascript-specials/article.md b/1-js/02-first-steps/18-javascript-specials/article.md similarity index 83% rename from 1-js/02-first-steps/16-javascript-specials/article.md rename to 1-js/02-first-steps/18-javascript-specials/article.md index 9de1d6251..e7ddacac4 100644 --- a/1-js/02-first-steps/16-javascript-specials/article.md +++ b/1-js/02-first-steps/18-javascript-specials/article.md @@ -53,9 +53,9 @@ To fully enable all features of modern JavaScript, we should start scripts with ... ``` -The directive must be at the top of a script or at the beginning of a function. +The directive must be at the top of a script or at the beginning of a function body. -Without `"use strict"`, everything still works, but some features behave in the old-fashion, "compatible" way. We'd generally prefer the modern behavior. +Without `"use strict"`, everything still works, but some features behave in the old-fashioned, "compatible" way. We'd generally prefer the modern behavior. Some modern features of the language (like classes that we'll study in the future) enable strict mode implicitly. @@ -81,9 +81,10 @@ let x = 5; x = "John"; ``` -There are 7 data types: +There are 8 data types: - `number` for both floating-point and integer numbers, +- `bigint` for integer numbers of arbitrary length, - `string` for strings, - `boolean` for logical values: `true/false`, - `null` -- a type with a single value `null`, meaning "empty" or "does not exist", @@ -102,13 +103,13 @@ More in: <info:variables> and <info:types>. We're using a browser as a working environment, so basic UI functions will be: -[`prompt(question, [default])`](mdn:api/Window/prompt) +[`prompt(question, [default])`](https://developer.mozilla.org/en-US/docs/Web/API/Window/prompt) : Ask a `question`, and return either what the visitor entered or `null` if they clicked "cancel". -[`confirm(question)`](mdn:api/Window/confirm) +[`confirm(question)`](https://developer.mozilla.org/en-US/docs/Web/API/Window/confirm) : Ask a `question` and suggest to choose between Ok and Cancel. The choice is returned as `true/false`. -[`alert(message)`](mdn:api/Window/alert) +[`alert(message)`](https://developer.mozilla.org/en-US/docs/Web/API/Window/alert) : Output a `message`. All these functions are *modal*, they pause the code execution and prevent the visitor from interacting with the page until they answer. @@ -143,13 +144,16 @@ Assignments : There is a simple assignment: `a = b` and combined ones like `a *= 2`. Bitwise -: Bitwise operators work with integers on bit-level: see the [docs](mdn:/JavaScript/Reference/Operators/Bitwise_Operators) when they are needed. +: Bitwise operators work with 32-bit integers at the lowest, bit-level: see the [docs](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Expressions_and_Operators#bitwise_operators) when they are needed. -Ternary +Conditional : The only operator with three parameters: `cond ? resultA : resultB`. If `cond` is truthy, returns `resultA`, otherwise `resultB`. Logical operators -: Logical AND `&&` and OR `||` perform short-circuit evaluation and then return the value where it stopped. Logical NOT `!` converts the operand to boolean type and returns the inverse value. +: Logical AND `&&` and OR `||` perform short-circuit evaluation and then return the value where it stopped (not necessary `true`/`false`). Logical NOT `!` converts the operand to boolean type and returns the inverse value. + +Nullish coalescing operator +: The `??` operator provides a way to choose a defined value from a list of variables. The result of `a ?? b` is `a` unless it's `null/undefined`, then `b`. Comparisons : Equality check `==` for values of different types converts them to a number (except `null` and `undefined` that equal each other and nothing else), so these are equal: @@ -161,7 +165,7 @@ Comparisons Other comparisons convert to a number as well. - The strict equality operator `===` doesn't do the conversion: different types always mean different values for it, so: + The strict equality operator `===` doesn't do the conversion: different types always mean different values for it. Values `null` and `undefined` are special: they equal `==` each other and don't equal anything else. @@ -170,7 +174,7 @@ Comparisons Other operators : There are few others, like a comma operator. -More in: <info:operators>, <info:comparison>, <info:logical-operators>. +More in: <info:operators>, <info:comparison>, <info:logical-operators>, <info:nullish-coalescing-operator>. ## Loops @@ -212,6 +216,7 @@ let age = prompt('Your age?', 18); switch (age) { case 18: alert("Won't work"); // the result of prompt is a string, not a number + break; case "18": alert("This works!"); @@ -245,15 +250,13 @@ We covered three ways to create a function in JavaScript: let result = a + b; return result; - } + }; ``` - Function expressions can have a name, like `sum = function name(a, b)`, but that `name` is only visible inside that function. - 3. Arrow functions: ```js - // expression at the right side + // expression on the right side let sum = (a, b) => a + b; // or multi-line syntax with { ... }, need return here: @@ -270,17 +273,11 @@ We covered three ways to create a function in JavaScript: ``` -- Functions may have local variables: those declared inside its body. Such variables are only visible inside the function. +- Functions may have local variables: those declared inside its body or its parameter list. Such variables are only visible inside the function. - Parameters can have default values: `function sum(a = 1, b = 2) {...}`. - Functions always return something. If there's no `return` statement, then the result is `undefined`. - -| Function Declaration | Function Expression | -|----------------------|---------------------| -| visible in the whole code block | created when the execution reaches it | -| - | can have a name, visible only inside the function | - -More: see <info:function-basics>, <info:function-expressions-arrows>. +Details: see <info:function-basics>, <info:arrow-functions-basics>. ## More to come diff --git a/1-js/03-code-quality/01-debugging-chrome/article.md b/1-js/03-code-quality/01-debugging-chrome/article.md index 6d5bbbed3..4f50fb428 100644 --- a/1-js/03-code-quality/01-debugging-chrome/article.md +++ b/1-js/03-code-quality/01-debugging-chrome/article.md @@ -1,44 +1,44 @@ -# Debugging in Chrome +# Debugging in the browser Before writing more complex code, let's talk about debugging. -All modern browsers and most other environments support "debugging" -- a special UI in developer tools that makes finding and fixing errors much easier. +[Debugging](https://en.wikipedia.org/wiki/Debugging) is the process of finding and fixing errors within a script. All modern browsers and most other environments support debugging tools -- a special UI in developer tools that makes debugging much easier. It also allows to trace the code step by step to see what exactly is going on. -We'll be using Chrome here, because it's probably the most feature-rich in this aspect. +We'll be using Chrome here, because it has enough features, most other browsers have a similar process. -## The "sources" pane +## The "Sources" panel Your Chrome version may look a little bit different, but it still should be obvious what's there. - Open the [example page](debugging/index.html) in Chrome. - Turn on developer tools with `key:F12` (Mac: `key:Cmd+Opt+I`). -- Select the `sources` pane. +- Select the `Sources` panel. Here's what you should see if you are doing it for the first time:  -The toggler button <span class="devtools" style="background-position:-168px -76px"></span> opens the tab with files. +The toggler button <span class="devtools" style="background-position:-172px -98px"></span> opens the tab with files. -Let's click it and select `index.html` and then `hello.js` in the tree view. Here's what should show up: +Let's click it and select `hello.js` in the tree view. Here's what should show up:  -Here we can see three zones: +The Sources panel has 3 parts: -1. The **Resources zone** lists HTML, JavaScript, CSS and other files, including images that are attached to the page. Chrome extensions may appear here too. -2. The **Source zone** shows the source code. -3. The **Information and control zone** is for debugging, we'll explore it soon. +1. The **File Navigator** pane lists HTML, JavaScript, CSS and other files, including images that are attached to the page. Chrome extensions may appear here too. +2. The **Code Editor** pane shows the source code. +3. The **JavaScript Debugging** pane is for debugging, we'll explore it soon. -Now you could click the same toggler <span class="devtools" style="background-position:-200px -76px"></span> again to hide the resources list and give the code some space. +Now you could click the same toggler <span class="devtools" style="background-position:-172px -122px"></span> again to hide the resources list and give the code some space. ## Console -If we press `Esc`, then a console opens below. We can type commands there and press `key:Enter` to execute. +If we press `key:Esc`, then a console opens below. We can type commands there and press `key:Enter` to execute. After a statement is executed, its result is shown below. -For example, here `1+2` results in `3`, and `hello("debugger")` returns nothing, so the result is `undefined`: +For example, here `1+2` results in `3`, while the function call `hello("debugger")` returns nothing, so the result is `undefined`:  @@ -56,21 +56,21 @@ A *breakpoint* is a point of code where the debugger will automatically pause th While the code is paused, we can examine current variables, execute commands in the console etc. In other words, we can debug it. -We can always find a list of breakpoints in the right pane. That's useful when we have many breakpoints in various files. It allows us to: -- Quickly jump to the breakpoint in the code (by clicking on it in the right pane). +We can always find a list of breakpoints in the right panel. That's useful when we have many breakpoints in various files. It allows us to: +- Quickly jump to the breakpoint in the code (by clicking on it in the right panel). - Temporarily disable the breakpoint by unchecking it. - Remove the breakpoint by right-clicking and selecting Remove. - ...And so on. ```smart header="Conditional breakpoints" -*Right click* on the line number allows to create a *conditional* breakpoint. It only triggers when the given expression is truthy. +*Right click* on the line number allows to create a *conditional* breakpoint. It only triggers when the given expression, that you should provide when you create it, is truthy. That's handy when we need to stop only for a certain variable value or for certain function parameters. ``` -## Debugger command +## The command "debugger" -We can also pause the code by using the `debugger` command, like this: +We can also pause the code by using the `debugger` command in it, like this: ```js function hello(name) { @@ -84,12 +84,11 @@ function hello(name) { } ``` -That's very convenient when we are in a code editor and don't want to switch to the browser and look up the script in developer tools to set the breakpoint. - +Such command works only when the development tools are open, otherwise the browser ignores it. ## Pause and look around -In our example, `hello()` is called during the page load, so the easiest way to activate the debugger is to reload the page. So let's press `key:F5` (Windows, Linux) or `key:Cmd+R` (Mac). +In our example, `hello()` is called during the page load, so the easiest way to activate the debugger (after we've set the breakpoints) is to reload the page. So let's press `key:F5` (Windows, Linux) or `key:Cmd+R` (Mac). As the breakpoint is set, the execution pauses at the 4th line: @@ -99,13 +98,13 @@ Please open the informational dropdowns to the right (labeled with arrows). They 1. **`Watch` -- shows current values for any expressions.** - You can click the plus `+` and input an expression. The debugger will show its value at any moment, automatically recalculating it in the process of execution. + You can click the plus `+` and input an expression. The debugger will show its value, automatically recalculating it in the process of execution. 2. **`Call Stack` -- shows the nested calls chain.** At the current moment the debugger is inside `hello()` call, called by a script in `index.html` (no function there, so it's called "anonymous"). - If you click on a stack item, the debugger jumps to the corresponding code, and all its variables can be examined as well. + If you click on a stack item (e.g. "anonymous"), the debugger jumps to the corresponding code, and all its variables can be examined as well. 3. **`Scope` -- current variables.** `Local` shows local function variables. You can also see their values highlighted right over the source. @@ -118,52 +117,65 @@ Please open the informational dropdowns to the right (labeled with arrows). They Now it's time to *trace* the script. -There are buttons for it at the top of the right pane. Let's engage them. - -<span class="devtools" style="background-position:-7px -76px"></span> -- continue the execution, hotkey `key:F8`. +There are buttons for it at the top of the right panel. Let's engage them. +<!-- https://github.com/ChromeDevTools/devtools-frontend/blob/master/front_end/Images/src/largeIcons.svg --> +<span class="devtools" style="background-position:-146px -168px"></span> -- "Resume": continue the execution, hotkey `key:F8`. : Resumes the execution. If there are no additional breakpoints, then the execution just continues and the debugger loses control. Here's what we can see after a click on it:  - The execution has resumed, reached another breakpoint inside `say()` and paused there. Take a look at the "Call stack" at the right. It has increased by one more call. We're inside `say()` now. + The execution has resumed, reached another breakpoint inside `say()` and paused there. Take a look at the "Call Stack" at the right. It has increased by one more call. We're inside `say()` now. + +<span class="devtools" style="background-position:-200px -190px"></span> -- "Step": run the next command, hotkey `key:F9`. +: Run the next statement. If we click it now, `alert` will be shown. + + Clicking this again and again will step through all script statements one by one. + +<span class="devtools" style="background-position:-62px -192px"></span> -- "Step over": run the next command, but *don't go into a function*, hotkey `key:F10`. +: Similar to the previous "Step" command, but behaves differently if the next statement is a function call (not a built-in, like `alert`, but a function of our own). + + If we compare them, the "Step" command goes into a nested function call and pauses the execution at its first line, while "Step over" executes the nested function call invisibly to us, skipping the function internals. + + The execution is then paused immediately after that function call. + + That's good if we're not interested to see what happens inside the function call. -<span class="devtools" style="background-position:-137px -76px"></span> -- make a step (run the next command), but *don't go into the function*, hotkey `key:F10`. -: If we click it now, `alert` will be shown. The important thing is that `alert` can be any function, the execution "steps over it", skipping the function internals. +<span class="devtools" style="background-position:-4px -194px"></span> -- "Step into", hotkey `key:F11`. +: That's similar to "Step", but behaves differently in case of asynchronous function calls. If you're only starting to learn JavaScript, then you can ignore the difference, as we don't have asynchronous calls yet. -<span class="devtools" style="background-position:-72px -76px"></span> -- make a step, hotkey `key:F11`. -: The same as the previous one, but "steps into" nested functions. Clicking this will step through all script actions one by one. + For the future, just note that "Step" command ignores async actions, such as `setTimeout` (scheduled function call), that execute later. The "Step into" goes into their code, waiting for them if necessary. See [DevTools manual](https://developers.google.com/web/updates/2018/01/devtools#async) for more details. -<span class="devtools" style="background-position:-104px -76px"></span> -- continue the execution till the end of the current function, hotkey `key:Shift+F11`. -: The execution would stop at the very last line of the current function. That's handy when we accidentally entered a nested call using <span class="devtools" style="background-position:-72px -76px"></span>, but it does not interest us, and we want to continue to its end as soon as possible. +<span class="devtools" style="background-position:-32px -194px"></span> -- "Step out": continue the execution till the end of the current function, hotkey `key:Shift+F11`. +: Continue the execution and stop it at the very last line of the current function. That's handy when we accidentally entered a nested call using <span class="devtools" style="background-position:-200px -190px"></span>, but it does not interest us, and we want to continue to its end as soon as possible. -<span class="devtools" style="background-position:-7px -28px"></span> -- enable/disable all breakpoints. +<span class="devtools" style="background-position:-61px -74px"></span> -- enable/disable all breakpoints. : That button does not move the execution. Just a mass on/off for breakpoints. -<span class="devtools" style="background-position:-264px -4px"></span> -- enable/disable automatic pause in case of an error. -: When enabled, and the developer tools is open, a script error automatically pauses the execution. Then we can analyze variables to see what went wrong. So if our script dies with an error, we can open debugger, enable this option and reload the page to see where it dies and what's the context at that moment. +<span class="devtools" style="background-position:-90px -146px"></span> -- enable/disable automatic pause in case of an error. +: When enabled, if the developer tools is open, an error during the script execution automatically pauses it. Then we can analyze variables in the debugger to see what went wrong. So if our script dies with an error, we can open debugger, enable this option and reload the page to see where it dies and what's the context at that moment. ```smart header="Continue to here" Right click on a line of code opens the context menu with a great option called "Continue to here". -That's handy when we want to move multiple steps forward, but we're too lazy to set a breakpoint. +That's handy when we want to move multiple steps forward to the line, but we're too lazy to set a breakpoint. ``` ## Logging -To output something to console, there's `console.log` function. +To output something to console from our code, there's `console.log` function. For instance, this outputs values from `0` to `4` to console: ```js run // open console to see for (let i = 0; i < 5; i++) { - console.log("value", i); + console.log("value,", i); } ``` -Regular users don't see that output, it is in the console. To see it, either open the Console tab of developer tools or press `key:Esc` while in another tab: that opens the console at the bottom. +Regular users don't see that output, it is in the console. To see it, either open the Console panel of developer tools or press `key:Esc` while in another panel: that opens the console at the bottom. If we have enough logging in our code, then we can see what's going on from the records, without the debugger. @@ -172,12 +184,12 @@ If we have enough logging in our code, then we can see what's going on from the As we can see, there are three main ways to pause a script: 1. A breakpoint. 2. The `debugger` statements. -3. An error (if dev tools are open and the button <span class="devtools" style="background-position:-264px -4px"></span> is "on"). +3. An error (if dev tools are open and the button <span class="devtools" style="background-position:-90px -146px"></span> is "on"). -Then we can examine variables and step on to see where the execution goes wrong. +When paused, we can debug: examine variables and trace the code to see where the execution goes wrong. There are many more options in developer tools than covered here. The full manual is at <https://developers.google.com/web/tools/chrome-devtools>. The information from this chapter is enough to begin debugging, but later, especially if you do a lot of browser stuff, please go there and look through more advanced capabilities of developer tools. -Oh, and also you can click at various places of dev tools and just see what's showing up. That's probably the fastest route to learn dev tools. Don't forget about the right click as well! +Oh, and also you can click at various places of dev tools and just see what's showing up. That's probably the fastest route to learn dev tools. Don't forget about the right click and context menus! diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-open-sources.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-open-sources.svg index a3c7db6ec..a647276e8 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-open-sources.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-open-sources.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="228" viewBox="0 0 697 228"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v228H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-open-sources.svg"><g id="Bitmap"><image width="698" height="228" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="open-sources" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="28" y="123">open sources</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M28.662 53.632l19.852 7.56-5.89 5.413 32.98 35.88 1.015 1.104-2.208 2.03-1.015-1.104-32.981-35.88-5.89 5.414-5.863-20.417z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="228" viewBox="0 0 697 228"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v228H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-open-sources.svg"><g id="Bitmap"><image width="698" height="228" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="open-sources" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="28" y="123">open sources</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M28.662 53.632l19.852 7.56-5.89 5.413 32.98 35.88 1.015 1.104-2.208 2.03-1.015-1.104-32.981-35.88-5.89 5.414-5.863-20.417z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="228" viewBox="0 0 697 228"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v228H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-open-sources.svg"><g id="Bitmap"><image width="698" height="228" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="open-sources" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="28" y="123">open sources</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M28.662 53.632l19.852 7.56-5.89 5.413 32.98 35.88 1.015 1.104-2.208 2.03-1.015-1.104-32.981-35.88-5.89 5.414-5.863-20.417z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-breakpoint.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-breakpoint.svg index 6e7b60f85..d7aaabf02 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-breakpoint.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-breakpoint.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="244" viewBox="0 0 697 244"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h697v244H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-breakpoint.svg"><g id="Bitmap"><image width="697" height="244" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="here's-the-list" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="310" y="106">here's the list</tspan></text><text id="breakpoints" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="127" y="130">breakpoints</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M421.013 114.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552zM119.02 129.64l.84 2.88-1.44.42-61.239 17.828 2.237 7.682-20.898-3.81 15.587-14.433 2.236 7.681 61.238-17.828 1.44-.42z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M58.797 100.63l-1.299 7.894 60.746 9.996 1.48.243-.487 2.96-1.48-.243-60.747-9.996-1.298 7.894-17.205-12.46 20.29-6.288z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="244" viewBox="0 0 697 244"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h697v244H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-breakpoint.svg"><g id="Bitmap"><image width="697" height="244" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="here's-the-list" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="310" y="106">here's the list</tspan></text><text id="breakpoints" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="127" y="130">breakpoints</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M421.013 114.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552zM119.02 129.64l.84 2.88-1.44.42-61.239 17.828 2.237 7.682-20.898-3.81 15.587-14.433 2.236 7.681 61.238-17.828 1.44-.42z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M58.797 100.63l-1.299 7.894 60.746 9.996 1.48.243-.487 2.96-1.48-.243-60.747-9.996-1.298 7.894-17.205-12.46 20.29-6.288z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="697" height="244" viewBox="0 0 697 244"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h697v244H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-breakpoint.svg"><g id="Bitmap"><image width="697" height="244" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="here's-the-list" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="310" y="106">here's the list</tspan></text><text id="breakpoints" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="127" y="130">breakpoints</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M421.013 114.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552zM119.02 129.64l.84 2.88-1.44.42-61.239 17.828 2.237 7.682-20.898-3.81 15.587-14.433 2.236 7.681 61.238-17.828 1.44-.42z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M58.797 100.63l-1.299 7.894 60.746 9.996 1.48.243-.487 2.96-1.48-.243-60.747-9.996-1.298 7.894-17.205-12.46 20.29-6.288z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-console.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-console.svg index d5d2a0b93..36e7a2be6 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-console.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-console.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="693" height="156" viewBox="0 0 693 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h693v156H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-console.svg"><g id="Bitmap"><image width="693" height="156" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="Line-Copy-5-+-Line-Copy-4-+-Line-Copy-3" stroke="#C06334" stroke-linecap="square" stroke-width="3" transform="translate(172 59)"><path id="Line-Copy-5" d="M.5.5S25 .5 25 42.75.5 85 .5 85"/></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="693" height="156" viewBox="0 0 693 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h693v156H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-console.svg"><g id="Bitmap"><image width="693" height="156" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="Line-Copy-5-+-Line-Copy-4-+-Line-Copy-3" stroke="#C06334" stroke-linecap="square" stroke-width="3" transform="translate(172 59)"><path id="Line-Copy-5" d="M.5.5S25 .5 25 42.75.5 85 .5 85"/></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="693" height="156" viewBox="0 0 693 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h693v156H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-console.svg"><g id="Bitmap"><image width="693" height="156" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="Line-Copy-5-+-Line-Copy-4-+-Line-Copy-3" stroke="#C06334" stroke-linecap="square" stroke-width="3" transform="translate(172 59)"><path id="Line-Copy-5" d="M.5.5S25 .5 25 42.75.5 85 .5 85"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-pause.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-pause.svg index 83468fddb..3720509e7 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-pause.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-pause.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="700" height="304" viewBox="0 0 700 304"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="148">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="100">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="215">3</tspan></text><path id="rect-1" d="M0 0h700v304H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-pause.svg"><g id="Bitmap"><image width="700" height="304" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g><text id="see-the-outer-call-d" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="183" y="189">see the outer call details</tspan></text><text id="watch-expressions" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="238" y="99">watch expressions</tspan></text><text id="current-variables" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="249" y="214">current variables</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M452.5 84l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M452.5 175l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M452.5 199l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="700" height="304" viewBox="0 0 700 304"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="148">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="100">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="215">3</tspan></text><path id="rect-1" d="M0 0h700v304H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-pause.svg"><g id="Bitmap"><image width="700" height="304" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g><text id="see-the-outer-call-d" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="183" y="189">see the outer call details</tspan></text><text id="watch-expressions" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="238" y="99">watch expressions</tspan></text><text id="current-variables" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="249" y="214">current variables</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M452.5 84l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M452.5 175l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M452.5 199l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="700" height="304" viewBox="0 0 700 304"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="163.2%" height="127.9%" x="-31.6%" y="-14%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="148">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="100">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="591" y="215">3</tspan></text><path id="rect-1" d="M0 0h700v304H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-pause.svg"><g id="Bitmap"><image width="700" height="304" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g><text id="see-the-outer-call-d" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="183" y="189">see the outer call details</tspan></text><text id="watch-expressions" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="238" y="99">watch expressions</tspan></text><text id="current-variables" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="249" y="214">current variables</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M452.5 84l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M452.5 175l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M452.5 199l19 9.5-19 9.5v-8H412v-3h40.5v-8z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-trace-1.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-trace-1.svg index 23937e0d6..757a2578b 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-trace-1.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-sources-debugger-trace-1.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="698" height="310" viewBox="0 0 698 310"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v310H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-trace-1.svg"><g id="Bitmap"><image width="698" height="310" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="nested-calls" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="337" y="94">nested calls</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M412.013 99.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="698" height="310" viewBox="0 0 698 310"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v310H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-trace-1.svg"><g id="Bitmap"><image width="698" height="310" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="nested-calls" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="337" y="94">nested calls</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M412.013 99.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="698" height="310" viewBox="0 0 698 310"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v310H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-sources-debugger-trace-1.svg"><g id="Bitmap"><image width="698" height="310" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><text id="nested-calls" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="337" y="94">nested calls</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M412.013 99.435l1.276.79 39.263 24.271 4.207-6.804 11.166 18.07-21.156-1.91 4.205-6.804-39.263-24.272-1.276-.789 1.578-2.552z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/chrome-tabs.svg b/1-js/03-code-quality/01-debugging-chrome/chrome-tabs.svg index 41a3d8784..f0e9afd01 100644 --- a/1-js/03-code-quality/01-debugging-chrome/chrome-tabs.svg +++ b/1-js/03-code-quality/01-debugging-chrome/chrome-tabs.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="694" height="225" viewBox="0 0 694 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="337" y="206">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="69" y="206">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="570" y="207">3</tspan></text><path id="rect-1" d="M0 0h694v225H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-tabs.svg"><g id="Bitmap"><image width="694" height="225" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="694" height="225" viewBox="0 0 694 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="337" y="206">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="69" y="206">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="570" y="207">3</tspan></text><path id="rect-1" d="M0 0h694v225H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-tabs.svg"><g id="Bitmap"><image width="694" height="225" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="694" height="225" viewBox="0 0 694 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-3" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-5" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><filter id="filter-7" width="134.3%" height="114.6%" x="-17.1%" y="-7.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1.5" result="shadowSpreadOuter1"/><feOffset in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 1 0"/></filter><text id="text-2" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="337" y="206">2</tspan></text><text id="text-4" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="69" y="206">1</tspan></text><text id="text-6" font-family="OpenSans-Bold, Open Sans" font-size="60" font-weight="bold"><tspan x="570" y="207">3</tspan></text><path id="rect-1" d="M0 0h694v225H0z"/></defs><g id="debugging" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="chrome-tabs.svg"><g id="Bitmap"><image width="694" height="225" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="2" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-3)" xlink:href="#text-2"/><use xlink:href="#text-2"/><use xlink:href="#text-2"/></g><g id="1" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-5)" xlink:href="#text-4"/><use xlink:href="#text-4"/><use xlink:href="#text-4"/></g><g id="3" fill="#C06334" fill-rule="nonzero"><use filter="url(#filter-7)" xlink:href="#text-6"/><use xlink:href="#text-6"/><use xlink:href="#text-6"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/01-debugging-chrome/head.html b/1-js/03-code-quality/01-debugging-chrome/head.html index f219b0af1..615326c08 100644 --- a/1-js/03-code-quality/01-debugging-chrome/head.html +++ b/1-js/03-code-quality/01-debugging-chrome/head.html @@ -1,8 +1,8 @@ <style> span.devtools { display: inline-block; - background-image: url(/article/debugging-chrome/toolbarButtonGlyphs.svg); - height:16px; - width:16px; + background-image: url(/article/debugging-chrome/largeIcons.svg); + height:18px; + width:18px; } </style> diff --git a/1-js/03-code-quality/01-debugging-chrome/largeIcons.svg b/1-js/03-code-quality/01-debugging-chrome/largeIcons.svg new file mode 100644 index 000000000..83303365b --- /dev/null +++ b/1-js/03-code-quality/01-debugging-chrome/largeIcons.svg @@ -0,0 +1,1472 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:xlink="http://www.w3.org/1999/xlink" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="224" + height="216" + id="svg2" + version="1.1" + inkscape:version="0.92.2pre0 (973e216, 2017-07-25)" + sodipodi:docname="largeIcons.svg"> + <metadata + id="metadata606"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs604" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1946" + inkscape:window-height="1546" + id="namedview602" + showgrid="true" + inkscape:zoom="4.3703705" + inkscape:cx="92.781028" + inkscape:cy="118.13064" + inkscape:window-x="1047" + inkscape:window-y="228" + inkscape:window-maximized="0" + inkscape:current-layer="svg2" + inkscape:snap-grids="true" + showguides="true" + inkscape:guide-bbox="true"> + <inkscape:grid + type="xygrid" + id="grid3583" + empspacing="5" + visible="true" + enabled="true" + snapvisiblegridlinesonly="true" + spacingx="28" + spacingy="24" + originx="0" + originy="0" /> + </sodipodi:namedview> + <g + id="g4"> + <g + transform="translate(-322,-72)" + id="g8"> + <circle + transform="translate(326,74)" + cx="15" + cy="17" + r="3" + id="circle10" + sodipodi:cx="15" + sodipodi:cy="17" + sodipodi:rx="3" + sodipodi:ry="3" + style="fill:#009802" /> + <path + d="m 329,77 h 18 v 18 h -18 z" + id="path12" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + d="m 327.25,75.25 h 20 v 20 h -20 z" + id="path14" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + d="m 325.12,73.125 h 20 v 20 h -20 z" + id="path16" + inkscape:connector-curvature="0" + style="fill:none" /> + </g> + </g> + <g + transform="translate(26,0)" + id="g18"> + <path + transform="translate(-32,0)" + d="M 57,12 53.5,8 H 39 v 8 h 14.5" + id="path22" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2.003,24)" + id="g24"> + <path + transform="translate(-224,-120)" + d="m 240.77,127 h -1.534 v 4.233 h -4.233 v 1.534 h 4.233 V 137 h 1.534 v -4.233 h 4.233 v -1.534 H 240.77 V 127 z" + id="path28" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(26,24)" + id="g30"> + <path + transform="translate(-96,-145)" + d="m 103,148 h 18 v 18 h -18 z" + id="path34" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-96,-145)" + d="m 115.42,154.7 -6.705,-6.705 -1.0575,1.0575 1.785,1.785 -3.8625,3.8625 c -0.4425,0.4425 -0.4425,1.155 0,1.59 l 4.125,4.125 c 0.2175,0.2175 0.51,0.33 0.795,0.33 0.285,0 0.5775,-0.1125 0.795,-0.33 l 4.125,-4.125 c 0.4425,-0.435 0.4425,-1.1475 0,-1.59 z m -8.5125,0.795 3.5925,-3.5925 3.5925,3.5925 h -7.185 z m 10.342,1.125 c 0,0 -1.5,1.6275 -1.5,2.625 0,0.825 0.675,1.5 1.5,1.5 0.825,0 1.5,-0.675 1.5,-1.5 0,-0.9975 -1.5,-2.625 -1.5,-2.625 z" + id="path36" + inkscape:connector-curvature="0" /> + <path + transform="translate(-96,-145)" + d="m 103,163 h 18 v 3 h -18 z" + id="path38" + inkscape:connector-curvature="0" + style="fill-opacity:0.36000001" /> + </g> + <g + transform="translate(0,48)" + id="g40"> + <path + transform="translate(7,3)" + d="M 0,0 H 18 V 18 H 0 z" + id="path44" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(7,3)" + d="m 13,5 h 1.4936 C 15.32558,5 16,5.67154 16,6.5064 v 7.9871 c 0,0.83198 -0.67154,1.5064 -1.5064,1.5064 H 6.5065 c -0.83198,0 -1.5064,-0.67154 -1.5064,-1.5064 v -1.4936 h 6.4936 c 0.8349,0 1.5064,-0.67446 1.5064,-1.5064 V 4.9999 z" + id="path46" + inkscape:connector-curvature="0" + style="fill-opacity:0.36000001" /> + <path + transform="translate(7,3)" + d="M 3.5,2 C 2.669,2 2,2.669 2,3.5 v 8 C 2,12.331 2.669,13 3.5,13 h 8 c 0.831,0 1.5,-0.669 1.5,-1.5 v -8 C 13,2.669 12.331,2 11.5,2 h -8 z m 0,1.5 h 8 v 8 h -8 v -8 z" + id="path48" + inkscape:connector-curvature="0" + style="fill:#212121" /> + </g> + <g + transform="translate(26,48)" + id="g50"> + <path + transform="translate(-96,-24)" + d="m 107,33 h 8 v 6 h -8 z" + stroke-miterlimit="4.2" + id="path54" + inkscape:connector-curvature="0" + style="stroke:#000000;stroke-width:2;stroke-miterlimit:4.19999981" /> + <path + transform="translate(-96,-24)" + d="m 115,36 4,-4 v 8" + id="path56" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,0)" + id="g58"> + <g + id="g62" + style="stroke:#000000"> + <path + transform="matrix(0.36,0,0,0.36,-2.5,7.46)" + d="m 53,14 a 3,3 0 1 1 -6,0 3,3 0 1 1 6,0 z" + id="path64" + inkscape:connector-curvature="0" + style="fill-rule:evenodd;stroke-width:2.77800012" /> + <path + transform="translate(-128,-120)" + d="m 143.48,129.5 2.5403,-2 h -1.5242 v -2 h -2 l -0.0161,2 h -1.5403 z" + id="path66" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-120)" + d="m 146.5,132.5 2,2.5 v -1.4998 l 2,-1e-4 v -2.0002 l -2,1e-4 V 130 z" + id="path68" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-120)" + d="m 143.5,135.5 -2.5,2 h 1.5 v 2 h 2 v -2 h 1.5 z" + id="path70" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-120)" + d="m 140.5,132.5 -2,-2.5 v 1.4999 h -2 v 2.0002 l 2,-3e-4 V 135 z" + id="path72" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(56,24)" + id="g74"> + <g + id="g78" + style="fill-rule:evenodd"> + <path + transform="matrix(0.9018,0,0,0.9018,4.308,4.525)" + d="M 0,0 H 18 V 18 H 0 z" + id="path80" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="matrix(0.9018,0,0,0.9018,4.308,4.525)" + d="M 4.174,8.343 2.76,9.757 7.003,13.999 15.488,5.514 14.074,4.1 7.003,11.171 z" + id="path82" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(57,47)" + id="g84"> + <path + transform="translate(-68,-143)" + d="M 76.94,152 76,152.94 79.0533,156 76,159.06 l 0.94,0.94 4,-4 z" + id="path88" + inkscape:connector-curvature="0" /> + <path + transform="translate(-68,-143)" + d="M 80.94,152 80,152.94 83.0533,156 80,159.06 l 0.94,0.94 4,-4 z" + id="path90" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2,72)" + id="g92"> + <path + transform="translate(-64,0)" + d="m 80.44,16.94 c -2.48,0 -4.5,-2.02 -4.5,-4.5 0,-0.88 0.26,-1.7 0.69,-2.39 l 6.2,6.2 c -0.69,0.44 -1.51,0.69 -2.39,0.69 m 4.5,-4.5 c 0,0.88 -0.26,1.7 -0.69,2.39 l -6.2,-6.2 c 0.69,-0.44 1.51,-0.69 2.39,-0.69 2.48,0 4.5,2.02 4.5,4.5 M 80.5,6 C 76.91,6 74,8.91 74,12.5 74,16.09 76.91,19 80.5,19 84.09,19 87,16.09 87,12.5 87,8.91 84.09,6 80.5,6" + id="path96" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(28,72)" + id="g98"> + <g + id="g102" + style="fill-rule:evenodd"> + <path + transform="matrix(0.87153,0,0,0.87153,4.071,4.568)" + d="M 0,0 H 18 V 18 H 0 z" + id="path104" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="matrix(0.87153,0,0,0.87153,4.071,4.568)" + d="M 12,3.5 A 1.505,1.505 0 0 0 10.494,2 H 4.506 C 3.676,2 3,2.674 3,3.506 v 7.988 C 3,12.326 3.671,12.997 4.5,13 V 3.5 H 12 z M 6,6.506 C 6,5.674 6.676,5 7.506,5 h 5.988 C 14.326,5 15,5.672 15,6.506 v 7.988 C 15,15.326 14.324,16 13.494,16 H 7.506 A 1.505,1.505 0 0 1 6,14.494 V 6.506 z M 7.5,6.5 h 6 v 8 h -6 v -8 z" + id="path106" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(54,72)" + id="g108"> + <path + transform="translate(0,-24)" + d="M 25,36 21.36,32 H 19.57 L 14,40 h 7.36 L 25,36 z" + id="path112" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-24)" + d="m 7,32 v 8 H 9.05 L 14,32 H 7 z" + id="path114" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-24)" + d="M 9.67,44.55 8.08,43.23 18.84,27.45 20.43,28.77 9.67,44.55 z" + id="path116" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-24)" + d="M 0,24 H 24 V 48 H 0 z" + id="path118" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(0,-24)" + d="M 0,24 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z" + id="path120" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(0,-24)" + d="M 0,24 H 24 V 48 H 0 z" + id="path122" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(0,-24)" + d="M 0,24 H 24 V 48 H 0 z" + id="path124" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(0,-24)" + d="M 0,24 H 24 V 48 H 0 z" + id="path126" + inkscape:connector-curvature="0" + style="fill:none" /> + </g> + <g + transform="translate(82,0)" + id="g128"> + <path + transform="translate(-128,0)" + d="M 149,8.33 147.67,7 144,10.67 140.33,7 139,8.33 142.67,12 139,15.67 140.33,17 144,13.33 147.67,17 149,15.67 145.33,12 149,8.33 z" + id="path132" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,24)" + id="g134"> + <path + transform="translate(-32,-24)" + d="M 53,37 H 43 v -5 h 10 v 5 z M 41,42 H 55 V 30 H 41 v 12 z" + id="path138" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,48)" + id="g140"> + <path + transform="translate(-224,-48)" + d="m 238,64 h 7 v -8 h -7 z m 9,2 H 233 V 54 h 14 z" + id="path144" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,72)" + id="g146"> + <path + transform="translate(-256,-48)" + d="m 274,64 h -7 v -8 h 7 v 8 z m -9,2 h 14 V 54 h -14 v 12 z" + id="path150" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2,96)" + id="g152"> + <path + transform="translate(-160,0)" + d="m 169,16.089 v 2.9105 h 2.9105 L 180.1944,10.7156 177.2838,7.8051 169,16.089 z m 13.769,-7.9387 c 0.30785,-0.30785 0.30785,-0.79294 0,-1.1008 l -1.8191,-1.8191 c -0.30785,-0.30784 -0.79295,-0.30784 -1.1008,0 l -1.5206,1.5299 2.9105,2.9105 1.5299,-1.5206 z" + id="path156" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(26,96)" + id="g158"> + <path + transform="translate(-288,-120)" + d="m 310.77,127.04 -1.816,-1.8164 c -0.30331,-0.30338 -0.79716,-0.30338 -1.1005,0 l -2.4304,2.4309 -1.4894,-1.4935 -1.1005,1.1007 1.1044,1.1046 -6.9373,6.9348 v 3.695 h 3.6942 l 6.9373,-6.9387 1.1005,1.1046 1.1005,-1.1007 -1.4932,-1.4935 2.4304,-2.4309 c 0.3072,-0.30338 0.3072,-0.79345 0,-1.0968 z m -10.721,10.4 -1.4932,-1.4935 6.2724,-6.2736 1.4932,1.4935 -6.2723,6.2736 z" + id="path162" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,96)" + id="g164"> + <path + transform="translate(-32,-48)" + d="m 44,59 2,3 v 4 h 4 v -4 l 2,-3 z" + id="path168" + inkscape:connector-curvature="0" + style="opacity:0.5;fill:#424242" /> + <path + transform="translate(-32,-48)" + d="m 46.5,65.23 c 0.32,0.13 0.84,0.24 1.47,0.24 0.59,0 1.14,-0.1 1.53,-0.26 v -3.93 l 4,-4.57 v -0.19 h -11 v 0.22 l 4,4.57 v 3.93 z M 47.97,67 C 46.81,66.91 45.82,66.71 45,66.01 V 61.89 L 41,57.32 V 55 h 14 v 2.35 l -4,4.57 v 4.13 c -0.92,0.67 -2.1,0.94 -3.03,0.95" + id="path170" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82.001,96)" + id="g172"> + <path + transform="matrix(0.75,0,0,0.75,-74.421,-143.43)" + d="m 108.56,195.24 h 24 v 24 h -24 z" + id="path176" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="matrix(0.75,0,0,0.75,-74.421,-143.43)" + d="m 108.56,215.24 h 24 v 4 h -24 z" + id="path178" + inkscape:connector-curvature="0" + style="fill-opacity:0.36000001" /> + <path + transform="matrix(0.75,0,0,0.75,-74.421,-143.43)" + d="m 119.56,198.24 -5.5,14 h 2.25 l 1.12,-3 h 6.25 l 1.12,3 h 2.25 l -5.49,-14 h -2 z m -1.38,9 2.38,-6.33 2.38,6.33 h -4.76 z" + id="path180" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(106,1)" + id="g182"> + <path + transform="matrix(0,-1,1,0,-112,-260)" + d="m -278.5,126.5 h 14 v 12 h -14 z" + id="path186" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="matrix(0,1,1,0,-112,-260)" + d="m 272,132.5 -5,-3.25 v 6.5" + id="path188" + inkscape:connector-curvature="0" /> + <path + transform="matrix(0,-1,1,0,-112,-260)" + d="m -275,126 h 1 v 13 h -1 z" + id="path190" + inkscape:connector-curvature="0" /> + </g> + <path + style="fill:none;stroke:#000000" + inkscape:connector-curvature="0" + id="path196" + d="m 119.5,30.5 h 14 v 12 h -14 z" /> + <path + inkscape:connector-curvature="0" + id="path198" + d="m 126,36.5 5,-3.25 v 6.5" /> + <path + inkscape:connector-curvature="0" + id="path200" + d="m 123,30 h 1 v 13 h -1 z" /> + <g + transform="translate(111,48)" + id="g202"> + <path + transform="matrix(-1,0,0,1,-192,-120)" + d="m -214.5,126.5 h 14 v 12 h -14 z" + id="path206" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="translate(-192,-120)" + d="m 208,132.5 -5,-3.25 v 6.5" + id="path208" + inkscape:connector-curvature="0" /> + <path + transform="matrix(-1,0,0,1,-192,-120)" + d="m -211,126 h 1 v 13 h -1 z" + id="path210" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(106,73)" + id="g212"> + <path + transform="matrix(0,1,1,0,-88,283)" + d="m -278.5,102.5 h 14 v 12 h -14 z" + id="path216" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="matrix(0,-1,1,0,-88,283)" + d="m 272,108.5 -5,-3.25 v 6.5" + id="path218" + inkscape:connector-curvature="0" /> + <path + transform="matrix(0,1,1,0,-88,283)" + d="m -275,102 h 1 v 13 h -1 z" + id="path220" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(110,96)" + id="g222"> + <path + transform="translate(-224,0)" + d="m 237,11 h -4 V 7 h 4 v 4 z" + id="path226" + inkscape:connector-curvature="0" /> + <path + transform="translate(-224,0)" + d="m 247,9 h -9 V 7 h 9 v 2 z" + id="path228" + inkscape:connector-curvature="0" /> + <path + transform="translate(-224,0)" + d="m 247,11 h -9 v -1 h 9 v 1 z" + id="path230" + inkscape:connector-curvature="0" /> + <path + transform="translate(-224,0)" + d="m 247,15 h -9 v -2 h 9 v 2 z" + id="path232" + inkscape:connector-curvature="0" /> + <path + transform="translate(-224,0)" + d="m 237,17 h -4 v -4 h 4 v 4 z" + id="path234" + inkscape:connector-curvature="0" /> + <path + transform="translate(-224,0)" + d="m 247,17 h -9 v -1 h 9 v 1 z" + id="path236" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2,120)" + id="g238"> + <path + transform="translate(0,-144)" + d="m 23,157 h -2 v 2 h 2 z" + id="path242" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 23,161 h -2 v 2 c 1,0 2,-1 2,-2 z" + id="path244" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 23,153 h -2 v 2 h 2 z" + id="path246" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 21,149 v 2 h 2 c 0,-1 -1,-2 -2,-2 z" + id="path248" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 11,163 h 4 v -6 H 9 v 4 c 0,1.1 0.9,2 2,2 z" + id="path250" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="M 11,153 H 9 v 2 h 2 z" + id="path252" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 19,149 h -2 v 2 h 2 z" + id="path254" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 19,161 h -2 v 2 h 2 z" + id="path256" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 11,149 c -1,0 -2,1 -2,2 h 2 z" + id="path258" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-144)" + d="m 15,149 h -2 v 2 h 2 z" + id="path260" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(26,120)" + id="g262"> + <path + transform="translate(-290,-46)" + d="m 317,69 v -5 l -5,5 h 5 z" + id="path266" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,120)" + id="g268"> + <g + id="g272" + style="fill-rule:evenodd"> + <path + transform="matrix(1.4142,0,0,1.4142,-278.88,-36.772)" + d="m 209.92,31.305 a 1.0607,1.0607 0 1 1 -2.1213,0 1.0607,1.0607 0 1 1 2.1213,0 z" + id="path274" + inkscape:connector-curvature="0" /> + <path + transform="matrix(1.4142,0,0,1.4142,-278.88,-31.772)" + d="m 209.92,31.305 a 1.0607,1.0607 0 1 1 -2.1213,0 1.0607,1.0607 0 1 1 2.1213,0 z" + id="path276" + inkscape:connector-curvature="0" /> + <path + transform="matrix(1.4142,0,0,1.4142,-278.88,-26.772)" + d="m 209.92,31.305 a 1.0607,1.0607 0 1 1 -2.1213,0 1.0607,1.0607 0 1 1 2.1213,0 z" + id="path278" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(84,120)" + id="g280"> + <path + transform="translate(-162,-144)" + d="m 166,147 h 18 v 18 h -18 z" + id="path284" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-162,-144)" + d="m 181.52,154.53 c -0.51,-2.58 -2.7862,-4.5262 -5.5162,-4.5262 -2.1675,0 -4.0462,1.23 -4.9875,3.0262 -2.2538,0.24375 -4.0125,2.1525 -4.0125,4.4738 0,2.4862 2.0138,4.5 4.5,4.5 h 9.75 c 2.07,0 3.75,-1.68 3.75,-3.75 0,-1.98 -1.5412,-3.585 -3.4838,-3.7238 z m -0.26625,5.9738 h -9.75 c -1.6575,0 -3,-1.3425 -3,-3 0,-1.6575 1.3425,-3 3,-3 h 0.5325 c 0.49125,-1.7288 2.0775,-3 3.9675,-3 2.28,0 4.125,1.845 4.125,4.125 v 0.375 h 1.125 c 1.2412,0 2.25,1.0088 2.25,2.25 0,1.2412 -1.0088,2.25 -2.25,2.25 z" + id="path286" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(112,120)" + id="g288"> + <path + transform="translate(-226,-72)" + d="m 235,76 c -0.55,0 -0.99,0.45 -0.99,1 L 234,91 c 0,0.55 0.44,1 1,1 h 10 c 0.55,0 1,-0.45 1,-1 V 81 l -5,-5 h -6 z m 6,5 v -4 l 4,4 h -4 z" + id="path292" + inkscape:connector-curvature="0" /> + <path + transform="translate(-226,-72)" + d="m 231,75 h 18 v 18 h -18 z" + id="path294" + inkscape:connector-curvature="0" + style="fill:none" /> + </g> + <g + transform="translate(140,0)" + id="g296"> + <path + transform="translate(-162,-24)" + d="m 169,29 h 18 v 18 h -18 z" + id="path300" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-162,-24)" + d="m 167.25,27.25 h 20 v 20 h -20 z" + id="path302" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-162,-24)" + d="m 165.12,25.125 h 20 v 20 h -20 z" + id="path304" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-162,-24)" + d="m 171,28 c -0.55,0 -1,0.45 -1,1 v 14 c 0,0.55 0.44,1 1,1 h 5.0938 c -0.0656,-0.32311 -0.0937,-0.65753 -0.0937,-1 0,-2.7614 2.2386,-5 5,-5 0.34247,0 0.67689,0.02816 1,0.09375 v -5.0938 l -5,-5 h -6 z m 6,1 4,4 h -4 v -4 z" + id="path306" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(140,24)" + id="g308"> + <path + transform="translate(-66,-120)" + d="m 71,124 h 18 v 18 H 71 z" + id="path312" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-66,-120)" + d="m 82,129 -2,-2 h -5 c -0.55,0 -1,0.45 -1,1 v 10 c 0,0.55 0.45,1 1,1 h 12 c 0.55,0 1,-0.45 1,-1 v -8 c 0,-0.55 -0.45,-1 -1,-1 h -5 z" + id="path314" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(140,48)" + id="g316"> + <path + transform="translate(-258,-144)" + d="m 278,150 h -10.5 c -0.8325,0 -1.5,0.675 -1.5,1.5 v 9 c 0,0.825 0.6675,1.5 1.5,1.5 H 278 c 0.825,0 1.5,-0.675 1.5,-1.5 v -9 c 0,-0.825 -0.6675,-1.5 -1.5,-1.5 z m 0,10.5 H 267.5 V 153 H 278 v 7.5 z" + id="path320" + inkscape:connector-curvature="0" + style="fill:#010101" /> + <path + transform="translate(-258,-144)" + d="m 262.97,147.07 h 18 v 18 h -18 z" + id="path322" + inkscape:connector-curvature="0" + style="fill:none" /> + </g> + <g + transform="translate(140,72)" + id="g324"> + <path + transform="translate(6,3.9)" + d="M 9,1 H 3.9954 C 3.45567,1 3,1.45078 3,2.0068 v 11.986 c 0,0.5569 0.44565,1.0068 0.9954,1.0068 h 8.0092 C 12.54433,14.9996 13,14.54882 13,13.9928 V 4.9996 l -4,-4 z m 3,4 H 9 V 2 l 3,3 z M 6,7 11,9.5 6,12 V 7 z" + id="path328" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(140,96)" + id="g330"> + <g + id="g334" + style="stroke:#000000"> + <path + transform="matrix(0.72907,0,0,0.72907,5.617,4.598)" + d="M 7.854,6.963 6.877,7.279 6.857,8.617 6.889,8.705 7.398,9.699 6.188,10.578 5.398,9.787 5.326,9.73 4.047,9.336 3.443,10.166 4.213,11.26 4.291,11.313 5.287,11.817 4.824,13.241 3.72,13.065 3.628,13.061 2.362,13.493 v 1.027 l 1.266,0.434 0.092,-0.004 1.104,-0.178 0.463,1.424 -0.996,0.506 -0.078,0.051 -0.77,1.094 0.604,0.83 1.279,-0.393 0.072,-0.059 0.789,-0.791 1.211,0.879 -0.51,0.996 -0.031,0.086 0.02,1.338 0.977,0.316 0.803,-1.068 0.025,-0.09 0.172,-1.104 h 1.498 l 0.172,1.104 0.025,0.09 0.803,1.068 0.977,-0.316 0.02,-1.338 -0.031,-0.086 -0.51,-0.996 1.211,-0.879 0.789,0.791 0.072,0.059 1.279,0.393 0.604,-0.83 -0.771,-1.094 -0.076,-0.051 -0.996,-0.506 0.461,-1.424 1.105,0.178 0.092,0.004 1.266,-0.434 v -1.027 l -1.266,-0.432 -0.092,0.004 -1.105,0.176 -0.461,-1.424 0.996,-0.504 0.076,-0.053 0.771,-1.094 L 15.159,9.336 13.88,9.731 13.808,9.788 13.019,10.579 11.808,9.7 12.318,8.706 12.349,8.618 12.329,7.28 11.352,6.964 10.549,8.034 10.524,8.122 10.352,9.227 H 8.854 L 8.682,8.121 8.656,8.03 7.854,6.963 m 1.748,3.398 a 3.621,3.645 0 0 1 3.621,3.645 3.621,3.645 0 0 1 -3.621,3.646 3.621,3.645 0 0 1 -3.619,-3.646 3.621,3.645 0 0 1 3.619,-3.645 z" + id="path336" + inkscape:connector-curvature="0" /> + <path + transform="matrix(0.72907,0,0,0.72907,5.227,3.617)" + d="M 14.885,1.563 14.178,1.957 14.365,3.24 14.396,3.279 15.14,4.04 14.553,5.03 13.529,4.736 13.479,4.729 12.262,5.172 12.25,5.98 13.455,6.459 13.506,6.453 14.537,6.189 15.1,7.191 14.332,7.932 14.301,7.971 14.08,9.248 14.771,9.662 15.787,8.859 15.807,8.811 16.1,7.787 17.242,7.803 17.502,8.834 17.52,8.883 18.514,9.715 19.219,9.32 19.03,8.04 19,7.998 18.258,7.234 18.844,6.25 19.869,6.541 19.92,6.551 21.14,6.105 21.15,5.297 19.943,4.818 19.893,4.824 18.861,5.088 18.301,4.086 19.07,3.346 19.1,3.307 19.322,2.029 18.627,1.615 17.609,2.418 17.59,2.467 17.303,3.49 16.16,3.475 15.896,2.443 15.879,2.395 14.885,1.563 z m 1.814,2.138 a 1.938,1.938 0 0 1 1.938,1.938 1.938,1.938 0 0 1 -1.938,1.937 1.938,1.938 0 0 1 -1.937,-1.937 1.938,1.938 0 0 1 1.937,-1.938 z" + id="path338" + inkscape:connector-curvature="0" + style="stroke-width:0.89899999" /> + </g> + </g> + <g + transform="translate(138,120)" + id="g340"> + <g + id="g344" + style="fill:none"> + <path + transform="translate(8,4)" + d="M 0,0 H 16 V 16 H 0 V 0 z" + id="path346" + inkscape:connector-curvature="0" + style="opacity:0.5" /> + <path + transform="translate(8,4)" + d="M 6,14 H 3.5 C 2.5,14 2,13.5 2,12.5 v -9 C 2,2.5 2.5,2 3.5,2 h 9 C 14,2 14,3.4678 14,3.5 V 6 H 13 V 3 H 3 v 10 h 3 v 1 z m 9,-5 -3,2 3,3 -1,1 -3,-3 -2,3 -2,-8 8,2 z" + id="path348" + inkscape:connector-curvature="0" + style="fill:#000000" /> + </g> + </g> + <g + transform="translate(0,144)" + id="g350"> + <path + transform="translate(-98,-120)" + d="m 110.5,127.5 h -1 l 2,-2 2,2 h -1 c 0,0 -0.0345,4.6379 -0.0345,4.0345 l 4.0345,-0.034 v -1 l 2,2 -2,2 v -1 l -4.0345,-0.034 0.0345,4.0346 h 1 l -2,2 -2,-2 h 1 l 0.0345,-4.0346 -4.0345,0.034 v 1 l -2,-2 2,-2 v 1 l 4.0345,0.034 z" + id="path354" + inkscape:connector-curvature="0" + style="stroke:#000000" /> + </g> + <g + transform="translate(26,144)" + id="g356"> + <path + transform="translate(-320,0)" + d="m 333.6,15.2 h 1.6 V 8.8 h -1.6 v 6.4 z M 336,4 c -4.42,0 -8,3.58 -8,8 0,4.42 3.58,8 8,8 4.42,0 8,-3.58 8,-8 0,-4.42 -3.58,-8 -8,-8 z m 0,14.4 c -3.528,0 -6.4,-2.872 -6.4,-6.4 0,-3.528 2.872,-6.4 6.4,-6.4 3.528,0 6.4,2.872 6.4,6.4 0,3.528 -2.872,6.4 -6.4,6.4 z m 0.8,-3.2 h 1.6 V 8.8 h -1.6 v 6.4 z" + id="path360" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,144)" + id="g362"> + <path + transform="translate(-32,-72)" + d="m 47,88 h -3 v -8 h 3 v 8 z" + id="path366" + inkscape:connector-curvature="0" /> + <path + transform="translate(-32,-72)" + d="m 53,88 h -3 v -8 h 3 v 8 z" + id="path368" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,144)" + id="g370"> + <path + transform="translate(-256,0)" + d="m 275,15 h -2 V 9 h 2 z m -4,0 h -2 V 9 h 2 z m 4,-10 h -6 l -4,4 v 6 l 4,4 h 6 l 4,-4.12 V 9 z" + id="path374" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(110,144)" + id="g376"> + <g + font-weight="400" + id="g380" + style="font-weight:400;font-family:Sans"> + <path + transform="translate(-289,-96)" + d="m 300,101 v 4.0001 1.0001 l 3.3434,0.53115 -0.0156,8.0468 -1.3278,0.42183 c 0.006,0.59278 0.43931,1.0114 1,1 h 8 c 0.57896,0.002 0.98177,-0.42708 1,-1 v -14 c -0.0102,-0.53477 -0.48177,-0.99739 -1,-1 h -10 c -0.53297,0.008 -0.99716,0.45677 -1,1 z m 1,0 h 10 v 14 h -8 l 0.71845,-0.42179 0.0937,-8.4218 -2.8122,-0.1561 v -0.99993 z" + overflow="visible" + style="text-indent:0;line-height:normal;text-transform:none;block-progression:tb;overflow:visible" + id="path382" + inkscape:connector-curvature="0" /> + <path + transform="translate(-289,-96)" + d="m 297,106 v 9 c 0.006,0.59278 0.43931,1.0114 1,1 h 5 c 0.57896,0.002 0.98177,-0.42708 1,-1 v -9 c -0.0102,-0.53477 -0.48177,-0.99739 -1,-1 h -5 c -0.53297,0.008 -0.99716,0.45677 -1,1 z m 1,1.0002 h 5 v 7 h -5 z" + overflow="visible" + style="text-indent:0;line-height:normal;text-transform:none;block-progression:tb;overflow:visible" + id="path384" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(138,144)" + id="g386"> + <path + transform="translate(-320,-48)" + d="m 328,60 c 0,4.42 3.58,8 8,8 4.42,0 8,-3.58 8,-8 0,-4.42 -3.58,-8 -8,-8 -4.42,0 -8,3.58 -8,8 z m 8,6.4 c -3.528,0 -6.4,-2.872 -6.4,-6.4 0,-3.528 2.872,-6.4 6.4,-6.4 3.528,0 6.4,2.872 6.4,6.4 0,3.528 -2.872,6.4 -6.4,6.4 z M 334.5067,63.4286 339.0781,60 334.5067,56.5714 v 6.8571 z" + id="path390" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2,168)" + id="g392"> + <path + transform="translate(-96,-48)" + d="m 108,60 10,-5 v 10" + id="path396" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(26,168)" + id="g398"> + <path + transform="translate(-64,-48)" + d="M 86,60 76,55 v 10" + id="path402" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,168)" + id="g404"> + <path + transform="translate(-256,-24)" + d="m 268.01,35.99 c 0.61,0.28 1.07,0.9 1.07,1.58 0.11,0.85 -0.05,1.72 0.12,2.57 0.27,0.54 1,0.28 1.43,0.55 0.49,0.24 0.48,1.01 -0.06,1.18 -0.56,0.22 -1.18,0.08 -1.74,-0.05 -0.71,-0.2 -1.41,-0.72 -1.5,-1.5 -0.18,-0.89 0.01,-1.8 -0.16,-2.68 -0.22,-0.64 -0.94,-0.9 -1.57,-0.93 -0.58,-0.1 -0.83,-0.94 -0.35,-1.3 0.51,-0.35 1.26,-0.14 1.69,-0.66 0.44,-0.48 0.29,-1.18 0.32,-1.78 0,-0.81 -0.02,-1.77 0.65,-2.34 0.66,-0.54 1.58,-0.71 2.41,-0.63 0.63,0 0.98,0.87 0.4,1.22 -0.44,0.37 -1.2,0.06 -1.51,0.65 -0.14,0.56 -0.05,1.15 -0.07,1.73 -0.01,0.75 -0.05,1.64 -0.72,2.13 -0.12,0.1 -0.26,0.19 -0.4,0.26" + id="path408" + inkscape:connector-curvature="0" /> + <path + transform="translate(-256,-24)" + d="m 276.98,35.99 c -0.67,-0.3 -1.08,-1.02 -1.08,-1.75 -0.07,-0.76 0.03,-1.52 -0.06,-2.28 -0.24,-0.58 -0.98,-0.4 -1.46,-0.59 -0.59,-0.24 -0.48,-1.18 0.14,-1.31 0.73,-0.15 1.52,-0.01 2.18,0.32 0.56,0.28 0.95,0.86 0.99,1.48 0.13,0.83 -0.03,1.68 0.13,2.5 0.2,0.68 0.94,0.83 1.54,0.9 0.56,0.07 0.86,0.8 0.46,1.21 -0.44,0.46 -1.2,0.2 -1.65,0.66 -0.51,0.46 -0.4,1.21 -0.4,1.83 -0.03,0.78 0.06,1.69 -0.52,2.3 -0.74,0.65 -1.8,0.86 -2.75,0.68 -0.52,-0.16 -0.69,-1.01 -0.15,-1.25 0.44,-0.23 1.02,-0.08 1.41,-0.45 0.26,-0.45 0.09,-0.98 0.14,-1.47 0.01,-0.76 -0.07,-1.63 0.43,-2.26 0.18,-0.21 0.42,-0.37 0.66,-0.51" + id="path410" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,168)" + id="g412"> + <g + id="g416"> + <path + d="M 0,0 H 24 V 24 H 0 z" + id="path418" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + d="m 16,6 c -3.31,0 -6,2.69 -6,6 0,3.31 2.69,6 6,6 3.31,0 6,-2.69 6,-6 h -2 c 0,2.2091 -1.7909,4 -4,4 -2.2091,0 -4,-1.7909 -4,-4 0,-2.2091 1.7909,-4 4,-4 1.2756,0 2.3926,0.60127 3.125,1.5312 h 2.3438 C 20.52659,7.4496 18.4308,6 16,6 z" + id="path420" + inkscape:connector-curvature="0" /> + <path + d="M 21.091,6.88 V 9.9718 H 17.9992 z" + id="path422" + inkscape:connector-curvature="0" + style="stroke:#000000;stroke-width:1.06799996" /> + </g> + </g> + <g + transform="translate(110,168)" + id="g424"> + <path + transform="translate(8,4)" + d="M 2.5858,13.891 C 0.9959,12.4291 0,10.3315 0,8.0002 c 0,-4.42 3.58,-8 8,-8 4.42,0 8,3.58 8,8 0,4.42 -3.58,8 -8,8 v -1.6 c 3.528,0 6.4,-2.872 6.4,-6.4 0,-3.528 -2.872,-6.4 -6.4,-6.4 -3.528,0 -6.4,2.872 -6.4,6.4 0,1.9357 0.86461,3.674 2.2278,4.8487 l 1.6925,-1.4201 0.015145,4.463 -4.3925,-0.7899 1.443,-1.2108 z M 6.5066,11.4287 11.078,8.0001 6.5066,4.5715 v 6.8571 z" + id="path428" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(138,168)" + id="g430"> + <path + transform="translate(0,-72)" + d="M 23,84 15,79 V 89" + id="path434" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-72)" + d="M 13,89 H 10 V 79 h 3 v 10 z" + id="path436" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,0)" + id="g438"> + <path + transform="translate(-160,-120)" + d="m 169.2,130.15 c 0.0551,-0.1128 0.15339,-0.22 0.2313,-0.3064 0.54584,-0.6074 1.3889,-1.1974 3.0053,-1.5702 v -2.1715 c -1.2931,0.2523 -2.3146,0.6686 -3.304,1.5597 -0.33248,0.3832 -0.50543,0.6247 -0.514,1.1665 -0.007,0.443 0.17732,0.8863 0.58141,1.3219 z m 6.6309,3.1292 c -1.6164,-0.024 -3.1091,-0.2558 -4.6228,-0.764 -2.6938,-0.9037 -3.0286,-2.1697 -3.1703,-2.5931 -0.0958,1.6897 0.028,2.9573 0.0346,3.0648 0.092,1.5004 1.2918,2.6742 2.1575,3.1151 1.7796,0.9067 3.6614,1.3756 5.601,1.4151 v 2.4842 l 3.251,-4.6483 -3.251,-4.6486 v 2.5748 z m -0.32327,-5.2959 c 1.6164,0.01 3.4529,0.3384 5.2454,1.1131 0.73528,0.3177 1.3275,0.7904 1.8082,1.3048 0.33152,-0.1523 0.80672,-0.7308 0.74126,-1.2923 -0.16163,-1.3879 -1.6264,-2.1912 -1.8801,-2.3045 -1.7704,-0.7908 -3.9751,-1.0356 -5.9147,-1.0589 v -1.7442 l -2.0132,2.8787 2.0132,2.8783 v -1.775 z m 8.4236,1.8658 c -0.0824,0.6478 -0.82676,2.0349 -3.8978,2.9138 v 4.5247 c 1.1314,-0.3677 2.465,-1.2956 3.1787,-2.1988 0.2911,-0.3677 0.46906,-0.875 0.57736,-1.242 0.33216,-1.1229 0.18604,-3.3136 0.14175,-3.9977 z" + id="path442" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,24)" + id="g444"> + <path + transform="translate(8,4)" + d="m 10.624,2.47 c 1.9155,0.90417 3.2862,2.7533 3.4971,4.9467 h 0.87866 C 14.70102,3.8234 11.68426,1 7.99976,1 L 7.61315,1.0175 9.84495,3.24 10.62403,2.47 z M 6.9629,2.02083 c -0.34561,-0.34417 -0.90209,-0.34417 -1.2418,0 l -3.7255,3.71 c -0.34561,0.34417 -0.34561,0.89833 0,1.2367 l 7.041,7.0117 c 0.34561,0.34417 0.90209,0.34417 1.2418,0 l 3.7255,-3.71 c 0.34561,-0.34417 0.34561,-0.89833 0,-1.2367 l -7.041,-7.0117 z m 2.6946,11.34 -7.041,-7.0117 3.7255,-3.71 7.041,7.0117 -3.7255,3.71 z M 5.3755,13.53 C 3.46,12.63167 2.0893,10.7767 1.8784,8.5833 H 0.99974 C 1.29848,12.1766 4.31524,15 7.99974,15 L 8.38635,14.9825 6.15455,12.76 5.37547,13.53 z" + id="path448" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,48)" + id="g450"> + <path + transform="translate(-288,-72)" + d="m 293.96,74.013 h 20 v 20 h -20 z" + id="path454" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-288,-72)" + d="m 297,77 h 18 v 18 h -18 z" + id="path456" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-288,-72)" + d="m 295.25,75.25 h 20 v 20 h -20 z" + id="path458" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-288,-72)" + d="m 293.12,73.125 h 20 v 20 h -20 z" + id="path460" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-288,-72)" + d="m 288,72 h 24 v 24 h -24 z" + id="path462" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(-288,-72)" + d="m 309.35,84.686 c 0.0288,-0.224 0.0504,-0.448 0.0504,-0.686 0,-0.238 -0.0216,-0.462 -0.0504,-0.686 l 1.5184,-1.155 c 0.13673,-0.105 0.17272,-0.294 0.0863,-0.448 l -1.4393,-2.422 c -0.0863,-0.154 -0.28065,-0.21 -0.43897,-0.154 l -1.7919,0.7 c -0.37422,-0.28 -0.77722,-0.511 -1.2162,-0.686 l -0.27341,-1.855 C 305.77334,77.126 305.62223,77 305.44232,77 h -2.8785 c -0.17992,0 -0.33104,0.126 -0.35263,0.294 l -0.27346,1.855 c -0.43898,0.175 -0.84198,0.413 -1.2162,0.686 l -1.7919,-0.7 c -0.16551,-0.063 -0.35262,0 -0.43898,0.154 l -1.4393,2.422 c -0.0935,0.154 -0.0504,0.343 0.0863,0.448 l 1.5184,1.155 c -0.0287,0.224 -0.0504,0.455 -0.0504,0.686 0,0.231 0.0216,0.462 0.0504,0.686 l -1.5184,1.155 c -0.13673,0.105 -0.17271,0.294 -0.0863,0.448 l 1.4393,2.422 c 0.0863,0.154 0.28067,0.21 0.43898,0.154 l 1.7919,-0.7 c 0.37421,0.28 0.77721,0.511 1.2162,0.686 l 0.27346,1.855 c 0.0216,0.168 0.17271,0.294 0.35263,0.294 h 2.8785 c 0.17991,0 0.33103,-0.126 0.35263,-0.294 l 0.27345,-1.855 c 0.43898,-0.175 0.84198,-0.413 1.2162,-0.686 l 1.7919,0.7 c 0.16552,0.063 0.35263,0 0.43898,-0.154 l 1.4393,-2.422 c 0.0863,-0.154 0.0504,-0.343 -0.0863,-0.448 l -1.5184,-1.155 z m -5.3469,1.764 c -1.3889,0 -2.5187,-1.099 -2.5187,-2.45 0,-1.351 1.1298,-2.45 2.5187,-2.45 1.3889,0 2.5187,1.099 2.5187,2.45 0,1.351 -1.1298,2.45 -2.5187,2.45 z" + id="path464" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(162,73)" + id="g466"> + <path + transform="matrix(0,-1,1,0,-136,251)" + d="m 232.5,150.5 h 14 v 12 h -14 z" + id="path470" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="matrix(0,-1,1,0,-136,251)" + d="m 244,156.5 -5,-3.25 v 6.5" + id="path472" + inkscape:connector-curvature="0" /> + <path + transform="matrix(0,-1,1,0,-136,251)" + d="m 236,150 h 1 v 13 h -1 z" + id="path474" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,96)" + id="g476"> + <path + transform="translate(-160,-72)" + d="m 168.5,78.5 h 14 v 12 h -14 z" + id="path480" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="translate(-160,-72)" + d="m 180,84.5 -5,-3.25 v 6.5" + id="path482" + inkscape:connector-curvature="0" /> + <path + transform="translate(-160,-72)" + d="m 172,78 h 1 v 13 h -1 z" + id="path484" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,120)" + id="g486"> + <path + transform="matrix(-1,0,0,1,-192,-96)" + d="m -214.5,102.5 h 14 v 12 h -14 z" + id="path490" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="translate(-192,-96)" + d="m 203,108.5 5,-3.25 v 6.5" + id="path492" + inkscape:connector-curvature="0" /> + <path + transform="matrix(-1,0,0,1,-192,-96)" + d="m -211,102 h 1 v 13 h -1 z" + id="path494" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(162,145)" + id="g496"> + <path + transform="matrix(0,1,1,0,-88,219)" + d="m -214.5,102.5 h 14 v 12 h -14 z" + id="path500" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000" /> + <path + transform="matrix(0,-1,1,0,-88,219)" + d="m 203,108.5 5,-3.25 v 6.5" + id="path502" + inkscape:connector-curvature="0" /> + <path + transform="matrix(0,1,1,0,-88,219)" + d="m -211,102 h 1 v 13 h -1 z" + id="path504" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(166,168)" + id="g506"> + <path + transform="translate(-288,0)" + d="m 298,12 c 0,3.31 2.69,6 6,6 3.31,0 6,-2.69 6,-6 0,-3.31 -2.69,-6 -6,-6 -3.31,0 -6,2.69 -6,6" + id="path510" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(-2,192)" + id="g512"> + <path + transform="translate(-64,-72)" + d="m 78.5,89 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" + id="path516" + inkscape:connector-curvature="0" /> + <path + transform="translate(-64,-72)" + d="m 79,78 v 4 h -3 l 4.5,4 4.5,-4 h -3 v -4 h -3 z" + id="path518" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(26,192)" + id="g520"> + <path + transform="translate(-96,-72)" + d="m 110.5,89 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" + id="path524" + inkscape:connector-curvature="0" /> + <path + transform="translate(-96,-72)" + d="m 112.5,78 -4.5,4 h 3 v 4 h 3 v -4 h 3 l -4.5,-4 z" + id="path526" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(54,192)" + id="g528"> + <path + transform="translate(-128,-72)" + d="m 142,86.5 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" + id="path532" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-72)" + d="m 137.25,87.03 c 2.55,-8.43 11.4,-8.73 13.94,0" + id="path534" + inkscape:connector-curvature="0" + style="fill:none;stroke:#000000;stroke-width:2.5" /> + <path + transform="translate(-128,-72)" + d="m 151.68,89 -4.54,-2.76 6.68,-2.1" + id="path536" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(82,192)" + id="g538"> + <path + transform="translate(-288,-24)" + d="m 295,36 c 0,4.97 4.03,9 9,9 4.97,0 9,-4.03 9,-9 0,-4.97 -4.03,-9 -9,-9 -4.97,0 -9,4.03 -9,9" + id="path542" + inkscape:connector-curvature="0" + style="fill:url(#sprite60_a)" /> + <path + transform="translate(-288,-24)" + d="m 298,36 c 0,3.31 2.69,6 6,6 3.31,0 6,-2.69 6,-6 0,-3.31 -2.69,-6 -6,-6 -3.31,0 -6,2.69 -6,6" + id="path544" + inkscape:connector-curvature="0" /> + <defs + id="defs546"> + <radialGradient + id="sprite60_b" + cx="0" + cy="0" + r="1" + gradientTransform="matrix(18,0,0,-18,680,341)" + gradientUnits="userSpaceOnUse"> + <stop + offset="0" + id="stop549" /> + <stop + stop-opacity="0" + offset="1" + id="stop551" /> + </radialGradient> + <radialGradient + id="sprite60_a" + cx="0" + cy="0" + r="1" + gradientTransform="matrix(9,0,0,9,304,36)" + gradientUnits="userSpaceOnUse" + xlink:href="#sprite60_b" /> + </defs> + </g> + <g + transform="translate(110,192)" + id="g554"> + <g + id="g558"> + <path + transform="translate(7,3)" + d="M 0,0 H 18 V 18 H 0 z" + id="path560" + inkscape:connector-curvature="0" + style="fill:none" /> + <path + transform="translate(7,3)" + d="M 15.25,8 H 16 V 6.125 5 H 5 V 6.125 8 h 0.75 c 0,-0.82843 0.67157,-1.5 1.5,-1.5 h 2.5 v 7.25 c 0,0.82843 -0.67157,1.5 -1.5,1.5 V 16 h 1.875 0.75 1.875 v -0.75 c -0.82843,0 -1.5,-0.67157 -1.5,-1.5 V 6.5 h 2.5 c 0.82843,0 1.5,0.67157 1.5,1.5 z" + id="path562" + inkscape:connector-curvature="0" + style="fill-opacity:0.36000001" /> + <path + transform="translate(7,3)" + d="M 12.25,5 H 13 V 3.125 2 H 2 V 3.125 5 h 0.75 c 0,-0.82843 0.67157,-1.5 1.5,-1.5 h 2.5 v 7.25 c 0,0.82843 -0.67157,1.5 -1.5,1.5 V 13 h 1.875 0.75 1.875 v -0.75 c -0.82843,0 -1.5,-0.67157 -1.5,-1.5 V 3.5 h 2.5 c 0.82843,0 1.5,0.67157 1.5,1.5 z" + id="path564" + inkscape:connector-curvature="0" /> + </g> + </g> + <g + transform="translate(138,192)" + id="g566"> + <path + transform="translate(-128,-24)" + d="m 139.5,33 h 9 L 147,43 h -6" + id="path570" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-24)" + d="m 147.5,30 h -2 v -1 h -3 v 1 h -2 c -0.55,0 -1,0.48 -1,1 v 1 h 1 7 1 v -1 c 0,-0.52 -0.45,-1 -1,-1" + id="path572" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(168,192)" + id="g574"> + <path + transform="translate(0,-48)" + d="M 10,57 H 8 v 9 h 11 v -2 h -9 z" + id="path578" + inkscape:connector-curvature="0" /> + <path + transform="translate(0,-48)" + d="m 13,56 h 7 v 5 h -7 z m -2,-2 v 9 h 11 v -9 z" + id="path580" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(194,0)" + id="g582"> + <path + transform="translate(-96,0)" + d="m 112,6.6909 c -3.6364,0 -6.7418,2.2618 -8,5.4545 1.2582,3.1927 4.3636,5.4545 8,5.4545 3.6364,0 6.7418,-2.2618 8,-5.4545 -1.2582,-3.1927 -4.3636,-5.4545 -8,-5.4545 z m 0,9.0909 c -2.0073,0 -3.6364,-1.6291 -3.6364,-3.6364 0,-2.0073 1.6291,-3.6364 3.6364,-3.6364 2.0073,0 3.6364,1.6291 3.6364,3.6364 0,2.0073 -1.6291,3.6364 -3.6364,3.6364 z m 0,-5.8182 c -1.2073,0 -2.1818,0.97455 -2.1818,2.1818 0,1.2073 0.97455,2.1818 2.1818,2.1818 1.2073,0 2.1818,-0.97455 2.1818,-2.1818 0,-1.2073 -0.97455,-2.1818 -2.1818,-2.1818 z" + id="path586" + inkscape:connector-curvature="0" /> + </g> + <g + transform="translate(194,24)" + id="g588"> + <path + transform="translate(-128,-48)" + d="m 153,57 h -11 v -2 h 11 v 2 z" + id="path592" + inkscape:connector-curvature="0" + style="opacity:0.2" /> + <path + transform="translate(-128,-48)" + d="m 142,57 h -6 v -2 h 6 v 2 z" + id="path594" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-48)" + d="m 145,60 h -6 v -2 h 6 v 2 z" + id="path596" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-48)" + d="m 150,63 h -6 v -2 h 6 v 2 z" + id="path598" + inkscape:connector-curvature="0" /> + <path + transform="translate(-128,-48)" + d="m 152,66 h -6 v -2 h 6 v 2 z" + id="path600" + inkscape:connector-curvature="0" /> + </g> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="12.355932" + y="231.55931" + id="text4355"><tspan + sodipodi:role="line" + id="tspan4357" + x="12.355932" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">a</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="40.355934" + y="231.55931" + id="text4355-3"><tspan + sodipodi:role="line" + id="tspan4357-6" + x="40.355934" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">b</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="68.355934" + y="231.55931" + id="text4355-7"><tspan + sodipodi:role="line" + id="tspan4357-5" + x="68.355934" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">c</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="96.355934" + y="231.55931" + id="text4355-35"><tspan + sodipodi:role="line" + id="tspan4357-62" + x="96.355934" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">d</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="124.35593" + y="231.55931" + id="text4355-9"><tspan + sodipodi:role="line" + id="tspan4357-1" + x="124.35593" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">e</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="152.35593" + y="231.55931" + id="text4355-2"><tspan + sodipodi:role="line" + id="tspan4357-7" + x="152.35593" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">f</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="180.35593" + y="231.55931" + id="text4355-0"><tspan + sodipodi:role="line" + id="tspan4357-9" + x="180.35593" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">g</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="208.35593" + y="231.55931" + id="text4355-36"><tspan + sodipodi:role="line" + id="tspan4357-0" + x="208.35593" + y="231.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">h</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="207.55931" + id="text4355-6"><tspan + sodipodi:role="line" + id="tspan4357-2" + x="-15.644068" + y="207.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">1</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="183.55931" + id="text4355-61"><tspan + sodipodi:role="line" + id="tspan4357-8" + x="-15.644068" + y="183.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">2</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="159.55931" + id="text4355-79"><tspan + sodipodi:role="line" + id="tspan4357-20" + x="-15.644068" + y="159.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">3</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="135.55931" + id="text4355-23"><tspan + sodipodi:role="line" + id="tspan4357-75" + x="-15.644068" + y="135.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">4</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="111.55931" + id="text4355-92"><tspan + sodipodi:role="line" + id="tspan4357-28" + x="-15.644068" + y="111.55931" + style="font-size:12px;line-height:1.25;font-family:sans-serif">5</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="87.559311" + id="text4355-97"><tspan + sodipodi:role="line" + id="tspan4357-3" + x="-15.644068" + y="87.559311" + style="font-size:12px;line-height:1.25;font-family:sans-serif">6</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="63.559311" + id="text4355-612"><tspan + sodipodi:role="line" + id="tspan4357-93" + x="-15.644068" + y="63.559311" + style="font-size:12px;line-height:1.25;font-family:sans-serif">7</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="39.559311" + id="text4355-1"><tspan + sodipodi:role="line" + id="tspan4357-94" + x="-15.644068" + y="39.559311" + style="font-size:12px;line-height:1.25;font-family:sans-serif">8</tspan></text> + <text + xml:space="preserve" + style="font-style:normal;font-weight:normal;line-height:0%;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#b3b3b3;fill-opacity:1;stroke:none" + x="-15.644068" + y="15.559311" + id="text4355-78"><tspan + sodipodi:role="line" + id="tspan4357-4" + x="-15.644068" + y="15.559311" + style="font-size:12px;line-height:1.25;font-family:sans-serif">9</tspan></text> + <g + id="Page-1-6" + style="fill:none;stroke:none" + transform="translate(200,50)"> + <g + id="undo"> + <rect + height="20" + width="20" + y="0" + x="0" + id="bounds" + style="fill:#ffffff;fill-opacity:0" /> + <path + id="shape" + d="M 4.3431457,9.3431458 C 5.790861,7.8954305 7.790861,7 10,7 c 3.555275,0 6.568879,2.3191676 7.610506,5.527197 L 16.155367,13 15.658589,13 C 14.834916,10.669615 12.612438,9 10,9 8.3431458,9 6.8431457,9.6715729 5.7573593,10.757359 L 8,13 2,13 2,7 l 2.3431457,2.3431458 0,0 z" + inkscape:connector-curvature="0" + style="fill:#000000" /> + </g> + </g> + <path + style="fill:#000000" + sodipodi:nodetypes="ccccccccccccc" + inkscape:connector-curvature="0" + d="m 215,82 -3,0 0,-4 -4,0 0,4 -3,0 5,6 z m -10,7 0,1 10,0 0,-1 z" + id="path4295" /> + <path + style="fill:none" + inkscape:connector-curvature="0" + d="m 196,72 h 24 v 24 h -24 z" + id="path4297" /> + <path + style="fill:#000000" + d="m 210,102 -5,6 3,0 0,4 4,0 0,-4 3,0 z m -5,11 0,1 10,0 0,-1 z" + id="path4295-3" + inkscape:connector-curvature="0" + sodipodi:nodetypes="ccccccccccccc" /> + <rect + style="fill:#000000;" + id="square-icon" + width="8" + height="8" + x="206" + y="176" /> + <g + style="fill:#000000" + id="g4456" + transform="matrix(0.78548728,0,0,0.78548728,200.6822,123.63151)"> + <path + id="path4442" + d="M 15.5,14 H 14.71 L 14.43,13.73 C 15.41,12.59 16,11.11 16,9.5 16,5.91 13.09,3 9.5,3 5.91,3 3,5.91 3,9.5 3,13.09 5.91,16 9.5,16 c 1.61,0 3.09,-0.59 4.23,-1.57 L 14,14.71 v 0.79 l 5,4.99 1.49,-1.49 z m -6,0 C 7.01,14 5,11.99 5,9.5 5,7.01 7.01,5 9.5,5 11.99,5 14,7.01 14,9.5 14,11.99 11.99,14 9.5,14 Z" + inkscape:connector-curvature="0" /> + <path + id="path4444" + d="M 0,0 H 24 V 24 H 0 Z" + inkscape:connector-curvature="0" + style="fill:none" /> + </g> + <path + inkscape:connector-curvature="0" + id="path524-5" + d="m 211.71941,203.4233 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" /> + <g + transform="translate(190.16772,189.5042)" + id="g520-3"> + <path + d="m 20.83228,13.9958 -4,-4.5 v 3 h -4 v 3 h 4 v 3 z" + id="path526-7" + inkscape:connector-curvature="0" + sodipodi:nodetypes="cccccccc" /> + </g> +</svg> diff --git a/1-js/03-code-quality/01-debugging-chrome/toolbarButtonGlyphs.svg b/1-js/03-code-quality/01-debugging-chrome/toolbarButtonGlyphs.svg deleted file mode 100644 index 5bdf20a83..000000000 --- a/1-js/03-code-quality/01-debugging-chrome/toolbarButtonGlyphs.svg +++ /dev/null @@ -1,1035 +0,0 @@ -<?xml version="1.0" encoding="UTF-8" standalone="no"?> -<svg - xmlns:sketch="http://www.bohemiancoding.com/sketch/ns" - xmlns:dc="http://purl.org/dc/elements/1.1/" - xmlns:cc="http://creativecommons.org/ns#" - xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" - xmlns:svg="http://www.w3.org/2000/svg" - xmlns="http://www.w3.org/2000/svg" - xmlns:xlink="http://www.w3.org/1999/xlink" - xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" - xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" - height="168" - version="1.1" - width="352" - xml:space="preserve" - id="svg3395" - inkscape:version="0.48.4 r9939" - sodipodi:docname="toolbarButtonGlyphs.svg"><metadata - id="metadata3773"><rdf:RDF><cc:Work - rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type - rdf:resource="http://purl.org/dc/dcmitype/StillImage" /></cc:Work></rdf:RDF></metadata><sodipodi:namedview - showgrid="true" - id="namedview3397" - inkscape:zoom="2.8284271" - inkscape:cx="155.02536" - inkscape:cy="18.05" - inkscape:window-width="1920" - inkscape:window-height="1172" - inkscape:window-x="1920" - inkscape:window-y="0" - inkscape:window-maximized="1" - inkscape:current-layer="svg3395" - inkscape:snap-global="false" - showguides="true" - inkscape:guide-bbox="true"><inkscape:grid - dotted="false" - empspacing="2" - enabled="true" - id="grid3327" - snapvisiblegridlinesonly="true" - spacingx="32px" - spacingy="24px" - type="xygrid" - visible="true" /></sodipodi:namedview><defs - id="defs3400"><linearGradient - id="a"><stop - offset="0" - stop-color="#606eda" - id="stop3403" /><stop - offset="1" - stop-color="#021db2" - id="stop3405" /></linearGradient><linearGradient - id="b"><stop - offset="0" - stop-color="#e59290" - id="stop3408" /><stop - offset="1" - stop-color="#e99890" - id="stop3410" /></linearGradient><linearGradient - id="c"><stop - offset="0" - stop-color="#c0544f" - id="stop3413" /><stop - offset="1" - stop-color="#d08481" - id="stop3415" /></linearGradient><linearGradient - id="d"><stop - offset="0" - stop-color="#ffa801" - stop-opacity="0" - id="stop3418" /><stop - offset="1" - stop-color="#f0fb3d" - id="stop3420" /></linearGradient><linearGradient - id="e"><stop - offset="0" - stop-color="#ffbd00" - stop-opacity="0.65" - id="stop3423" /><stop - offset="1" - stop-color="#fff" - stop-opacity="0.91" - id="stop3425" /></linearGradient><linearGradient - id="f"><stop - offset="0" - stop-color="#00d600" - stop-opacity="0" - id="stop3428" /><stop - offset="1" - stop-color="#d8fc7b" - stop-opacity="0.81" - id="stop3430" /></linearGradient><linearGradient - id="g"><stop - offset="0" - stop-color="#00ba00" - id="stop3433" /><stop - offset="1" - stop-color="#fff" - stop-opacity="0.91" - id="stop3435" /></linearGradient><linearGradient - id="h"><stop - offset="0" - stop-color="#00a104" - id="stop3438" /><stop - offset="1" - stop-color="#00c605" - id="stop3440" /></linearGradient><linearGradient - id="i"><stop - offset="0" - stop-color="#f00" - stop-opacity="0" - id="stop3443" /><stop - offset="1" - stop-color="#f0cb68" - stop-opacity="0.71" - id="stop3445" /></linearGradient><linearGradient - id="j"><stop - offset="0" - stop-color="#e60000" - stop-opacity="0.65" - id="stop3448" /><stop - offset="1" - stop-color="#fff" - stop-opacity="0.91" - id="stop3450" /></linearGradient><linearGradient - id="k"><stop - offset="0" - stop-color="#a10000" - id="stop3453" /><stop - offset="1" - stop-color="#c60000" - id="stop3455" /></linearGradient><linearGradient - id="l"><stop - offset="0" - stop-color="#d7687d" - id="stop3458" /><stop - offset="1" - stop-color="#b21402" - id="stop3460" /></linearGradient><radialGradient - cx="0" - cy="0" - fx="0" - fy="0" - gradientTransform="matrix(18,0,0,-18,680,341)" - gradientUnits="userSpaceOnUse" - id="z" - r="1" - spreadMethod="pad"><stop - offset="0" - id="stop3463" /><stop - offset="1" - stop-opacity="0" - id="stop3465" /></radialGradient><radialGradient - cx="0" - cy="0" - fx="0" - fy="0" - gradientTransform="matrix(9,0,0,9,304,36)" - gradientUnits="userSpaceOnUse" - id="A" - r="1" - spreadMethod="pad" - xlink:href="#z" /><linearGradient - gradientTransform="matrix(0,-0.58333333,-0.58333333,0,118,110)" - gradientUnits="userSpaceOnUse" - id="m" - x1="0" - x2="24" - xlink:href="#l" - y1="0" - y2="0" /><linearGradient - gradientTransform="matrix(0,1,-1,0,206,0)" - gradientUnits="userSpaceOnUse" - id="n" - x1="96.5" - x2="109.5" - xlink:href="#b" - y1="103" - y2="103" /><linearGradient - gradientTransform="matrix(0,1,-1,0,207,-17)" - gradientUnits="userSpaceOnUse" - id="o" - x1="113" - x2="127" - xlink:href="#c" - y1="104" - y2="104" /><linearGradient - gradientTransform="matrix(0,-0.41666666,-0.41666666,0,218,106)" - gradientUnits="userSpaceOnUse" - id="p" - x1="0" - x2="24" - xlink:href="#l" - y1="0" - y2="0" /><linearGradient - gradientTransform="matrix(0.71428571,0,0,0.71428571,132.28571,37.714287)" - gradientUnits="userSpaceOnUse" - id="q" - x1="113" - x2="127" - xlink:href="#a" - y1="104" - y2="104" /><linearGradient - gradientTransform="matrix(0,1.3793103,-1.3008129,0,363.08462,-218.35335)" - gradientUnits="userSpaceOnUse" - id="r" - x1="227.875" - x2="235.125" - xlink:href="#k" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(0,-0.62931035,0.92682926,0,133.46895,244.52849)" - gradientUnits="userSpaceOnUse" - id="s" - x1="227.875" - x2="235.125" - xlink:href="#j" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(0,0.43965518,0.78048781,0,148.48744,2.1206316)" - gradientUnits="userSpaceOnUse" - id="t" - x1="227.875" - x2="235.125" - xlink:href="#i" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(-0.01506784,1.3792098,-1.3007182,-0.01421029,377.66542,-216.8212)" - gradientUnits="userSpaceOnUse" - id="u" - x1="227.875" - x2="235.125" - xlink:href="#h" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(0.00687469,-0.62926454,0.92676181,0.01012483,142.86511,243.44332)" - gradientUnits="userSpaceOnUse" - id="v" - x1="227.875" - x2="235.125" - xlink:href="#g" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(-0.00480288,0.43962318,0.780431,0.00852617,160.60265,1.1603253)" - gradientUnits="userSpaceOnUse" - id="w" - x1="227.875" - x2="235.125" - xlink:href="#f" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(0,-0.62931035,0.92682926,0,155.46895,244.52849)" - gradientUnits="userSpaceOnUse" - id="x" - x1="227.875" - x2="235.125" - xlink:href="#e" - y1="103.15625" - y2="103.15625" /><linearGradient - gradientTransform="matrix(0,0.43965518,0.78048781,0,170.44325,2.076437)" - gradientUnits="userSpaceOnUse" - id="y" - x1="227.875" - x2="235.125" - xlink:href="#d" - y1="103.15625" - y2="103.15625" /><linearGradient - inkscape:collect="always" - xlink:href="#l-0" - id="linearGradient3239" - gradientUnits="userSpaceOnUse" - gradientTransform="matrix(0,-0.41666666,-0.41666666,0,249.99999,117)" - x1="0" - y1="0" - x2="24" - y2="0" /><linearGradient - id="l-0"><stop - offset="0" - stop-color="#d7687d" - id="stop3458-8" /><stop - offset="1" - stop-color="#b21402" - id="stop3460-8" /></linearGradient></defs><path - d="M 57,12 53.5,8 39,8 l 0,8 14.5,0" - id="path3481" - inkscape:connector-curvature="0" /><path - d="M 25,36 21.36,32 19.57,32 14,40 21.36,40 25,36 z" - id="path3483" - inkscape:connector-curvature="0" /><path - d="M 7,32 7,40 9.05,40 14,32 7,32 z" - id="path3485" - inkscape:connector-curvature="0" /><path - d="M 9.67,44.55 8.08,43.23 18.84,27.45 20.43,28.77 9.67,44.55 z" - id="path3487" - inkscape:connector-curvature="0" /><path - d="m 16,5 0,2 c -2.76,0 -5,2.24 -5,5 0,2.76 2.24,5 5,5 2.76,0 5,-2.24 5,-5 l -1.84,0 c -0.04,1.72 -1.43,3.09 -3.16,3.09 -1.75,0 -3.16,-1.4 -3.16,-3.15 0,-1.75 1.41,-3.16 3.16,-3.16 L 16,11 20.5,8 16,5 z" - id="path3489" - inkscape:connector-curvature="0" /><path - d="m 53,37 -10,0 0,-5 10,0 0,5 z m -12,5 14,0 0,-12 -14,0 0,12 z" - id="path3491" - inkscape:connector-curvature="0" /><path - d="m 10,57 -2,0 0,9 11,0 0,-2 -9,0 z" - id="path3493" - inkscape:connector-curvature="0" /><path - d="m 13,56 7,0 0,5 -7,0 z m -2,-2 0,9 11,0 0,-9 z" - id="path3495" - inkscape:connector-curvature="0" /><path - d="m 80.44,16.94 c -2.48,0 -4.5,-2.02 -4.5,-4.5 0,-0.88 0.26,-1.7 0.69,-2.39 l 6.2,6.2 c -0.69,0.44 -1.51,0.69 -2.39,0.69 m 4.5,-4.5 c 0,0.88 -0.26,1.7 -0.69,2.39 l -6.2,-6.2 c 0.69,-0.44 1.51,-0.69 2.39,-0.69 2.48,0 4.5,2.02 4.5,4.5 M 80.5,6 C 76.91,6 74,8.91 74,12.5 74,16.09 76.91,19 80.5,19 84.09,19 87,16.09 87,12.5 87,8.91 84.09,6 80.5,6" - id="path3505" - inkscape:connector-curvature="0" /><path - d="m 117,65 -10,0 0,-10 10,0 0,10 z" - id="path3507" - inkscape:connector-curvature="0" /><path - d="M 149,8.33 147.67,7 144,10.67 140.33,7 139,8.33 142.67,12 139,15.67 140.33,17 144,13.33 147.67,17 149,15.67 145.33,12 149,8.33 z" - id="path3509" - inkscape:connector-curvature="0" /><path - d="m 44,59 2,3 0,4 4,0 0,-4 2,-3 z" - id="path3511" - inkscape:connector-curvature="0" - style="opacity:0.5;fill:#424242" /><path - d="m 46.5,65.23 c 0.32,0.13 0.84,0.24 1.47,0.24 0.59,0 1.14,-0.1 1.53,-0.26 l 0,-3.93 4,-4.57 0,-0.19 -11,0 0,0.22 4,4.57 0,3.93 z M 47.97,67 C 46.81,66.91 45.82,66.71 45,66.01 l 0,-4.12 -4,-4.57 0,-2.32 14,0 0,2.35 -4,4.57 0,4.13 c -0.92,0.67 -2.1,0.94 -3.03,0.95" - id="path3513" - inkscape:connector-curvature="0" /><path - d="M 86,60 76,55 76,65" - id="path3515" - inkscape:connector-curvature="0" /><path - d="m 153,57 -11,0 0,-2 11,0 0,2 z" - id="path3517" - inkscape:connector-curvature="0" - style="opacity:0.2" /><path - d="m 142,57 -6,0 0,-2 6,0 0,2 z" - id="path3519" - inkscape:connector-curvature="0" /><path - d="m 145,60 -6,0 0,-2 6,0 0,2 z" - id="path3521" - inkscape:connector-curvature="0" /><path - d="m 150,63 -6,0 0,-2 6,0 0,2 z" - id="path3523" - inkscape:connector-curvature="0" /><path - d="m 152,66 -6,0 0,-2 6,0 0,2 z" - id="path3525" - inkscape:connector-curvature="0" /><path - d="m 170,67 -2,0 0,-9 2,0 0,9 z" - id="path3527" - inkscape:connector-curvature="0" /><path - d="m 173,67 -2,0 0,-11 2,0 0,11 z" - id="path3529" - inkscape:connector-curvature="0" /><path - d="m 176,67 -2,0 0,-13 2,0 0,13 z" - id="path3531" - inkscape:connector-curvature="0" /><path - d="m 179,67 -2,0 0,-4 2,0 0,4 z" - id="path3533" - inkscape:connector-curvature="0" /><path - d="m 182,67 -2,0 0,-9 2,0 0,9 z" - id="path3535" - inkscape:connector-curvature="0" /><path - d="m 185,67 -2,0 0,-11 2,0 0,11 z" - id="path3537" - inkscape:connector-curvature="0" /><path - d="m 208,57 -10,0 0,-2 10,0 0,2 z" - id="path3539" - inkscape:connector-curvature="0" /><path - d="m 217,57 -5,0 0,-2 5,0 0,2 z" - id="path3541" - inkscape:connector-curvature="0" /><path - d="m 200,61 -2,0 0,-2 2,0 0,2 z" - id="path3543" - inkscape:connector-curvature="0" /><path - d="m 213,61 -10,0 0,-2 10,0 0,2 z" - id="path3545" - inkscape:connector-curvature="0" /><path - d="m 217,61 -2,0 0,-2 2,0 0,2 z" - id="path3547" - inkscape:connector-curvature="0" /><path - d="m 205,65 -7,0 0,-2 7,0 0,2 z" - id="path3549" - inkscape:connector-curvature="0" /><path - d="m 217,65 -8,0 0,-2 8,0 0,2 z" - id="path3551" - inkscape:connector-curvature="0" /><path - d="m 274,64 -7,0 0,-8 7,0 0,8 z m -9,2 14,0 0,-12 -14,0 0,12 z" - id="path3553" - inkscape:connector-curvature="0" /><path - d="m 298,12 c 0,3.31 2.69,6 6,6 3.31,0 6,-2.69 6,-6 0,-3.31 -2.69,-6 -6,-6 -3.31,0 -6,2.69 -6,6" - id="path3555" - inkscape:connector-curvature="0" /><path - d="m 295,36 c 0,4.97 4.03,9 9,9 l 0,0 c 4.97,0 9,-4.03 9,-9 l 0,0 c 0,-4.97 -4.03,-9 -9,-9 l 0,0 c -4.97,0 -9,4.03 -9,9" - id="path3557" - inkscape:connector-curvature="0" - style="fill:url(#A)" /><path - d="m 298,36 c 0,3.31 2.69,6 6,6 3.31,0 6,-2.69 6,-6 0,-3.31 -2.69,-6 -6,-6 -3.31,0 -6,2.69 -6,6" - id="path3559" - inkscape:connector-curvature="0" /><path - d="m 239,30 c -2.76,0 -5,2.24 -5,5 0,2.76 2.24,5 5,5 1.03,0 1.96,-0.31 2.75,-0.84 L 245.56,43 247,41.56 243.19,37.75 C 243.71,36.96 244,36.02 244,35 c 0,-2.76 -2.24,-5 -5,-5 z m 0,2 c 1.66,0 3,1.34 3,3 0,1.66 -1.34,3 -3,3 -1.66,0 -3,-1.34 -3,-3 0,-1.66 1.34,-3 3,-3 z" - id="path3561" - inkscape:connector-curvature="0" /><path - d="m 233.73,63.76 5.25,-3.23 5.32,2.23 4.95,-4.03" - stroke-miterlimit="10" - id="path3563" - inkscape:connector-curvature="0" - style="fill:none;stroke:#000000;stroke-width:1.48000002;stroke-miterlimit:10" /><path - d="m 231,67 0,-13 -1,0 0,14 21,0 0,-1 z" - id="path3565" - inkscape:connector-curvature="0" /><path - d="m 237,11 -4,0 0,-4 4,0 0,4 z" - id="path3567" - inkscape:connector-curvature="0" /><path - d="m 247,9 -9,0 0,-2 9,0 0,2 z" - id="path3569" - inkscape:connector-curvature="0" /><path - d="m 247,11 -9,0 0,-1 9,0 0,1 z" - id="path3571" - inkscape:connector-curvature="0" /><path - d="m 247,15 -9,0 0,-2 9,0 0,2 z" - id="path3573" - inkscape:connector-curvature="0" /><path - d="m 237,17 -4,0 0,-4 4,0 0,4 z" - id="path3575" - inkscape:connector-curvature="0" /><path - d="m 247,17 -9,0 0,-1 9,0 0,1 z" - id="path3577" - inkscape:connector-curvature="0" /><path - d="m 176,38.13 c -1.18,0 -2.13,-0.96 -2.13,-2.13 0,-1.18 0.96,-2.13 2.13,-2.13 1.18,0 2.13,0.96 2.13,2.13 0,1.18 -0.96,2.13 -2.13,2.13 m 8,-3.73 -2.93,-0.53 1.72,-2.39 -2.26,-2.26 -2.46,1.79 -0.47,-3 -3.2,0 -0.4,3.13 -2.53,-1.92 -2.26,2.26 1.99,2.59 -3.2,0.33 0,3.2 3.33,0.33 -2.12,2.59 2.26,2.26 2.53,-2.12 0.4,3.33 3.2,0 0.4,-3.2 2.53,1.99 2.26,-2.26 -1.79,-2.45 3,-0.47 0,-3.2 z" - id="path3579" - inkscape:connector-curvature="0" /><path - d="m 268.01,35.99 c 0.61,0.28 1.07,0.9 1.07,1.58 0.11,0.85 -0.05,1.72 0.12,2.57 0.27,0.54 1,0.28 1.43,0.55 0.49,0.24 0.48,1.01 -0.06,1.18 -0.56,0.22 -1.18,0.08 -1.74,-0.05 -0.71,-0.2 -1.41,-0.72 -1.5,-1.5 -0.18,-0.89 0.01,-1.8 -0.16,-2.68 -0.22,-0.64 -0.94,-0.9 -1.57,-0.93 -0.58,-0.1 -0.83,-0.94 -0.35,-1.3 0.51,-0.35 1.26,-0.14 1.69,-0.66 0.44,-0.48 0.29,-1.18 0.32,-1.78 0,-0.81 -0.02,-1.77 0.65,-2.34 0.66,-0.54 1.58,-0.71 2.41,-0.63 0.63,0 0.98,0.87 0.4,1.22 -0.44,0.37 -1.2,0.06 -1.51,0.65 -0.14,0.56 -0.05,1.15 -0.07,1.73 -0.01,0.75 -0.05,1.64 -0.72,2.13 -0.12,0.1 -0.26,0.19 -0.4,0.26" - id="path3581" - inkscape:connector-curvature="0" /><path - d="m 276.98,35.99 c -0.67,-0.3 -1.08,-1.02 -1.08,-1.75 -0.07,-0.76 0.03,-1.52 -0.06,-2.28 -0.24,-0.58 -0.98,-0.4 -1.46,-0.59 -0.59,-0.24 -0.48,-1.18 0.14,-1.31 0.73,-0.15 1.52,-0.01 2.18,0.32 0.56,0.28 0.95,0.86 0.99,1.48 0.13,0.83 -0.03,1.68 0.13,2.5 0.2,0.68 0.94,0.83 1.54,0.9 0.56,0.07 0.86,0.8 0.46,1.21 -0.44,0.46 -1.2,0.2 -1.65,0.66 -0.51,0.46 -0.4,1.21 -0.4,1.83 -0.03,0.78 0.06,1.69 -0.52,2.3 -0.74,0.65 -1.8,0.86 -2.75,0.68 -0.52,-0.16 -0.69,-1.01 -0.15,-1.25 0.44,-0.23 1.02,-0.08 1.41,-0.45 0.26,-0.45 0.09,-0.98 0.14,-1.47 0.01,-0.76 -0.07,-1.63 0.43,-2.26 0.18,-0.21 0.42,-0.37 0.66,-0.51" - id="path3583" - inkscape:connector-curvature="0" /><path - d="m 275,15 -2,0 0,-6 2,0 z m -4,0 -2,0 0,-6 2,0 z m 4,-10 -6,0 -4,4 0,6 4,4 6,0 4,-4.12 0,-5.88 z" - id="path3585" - inkscape:connector-curvature="0" /><path - d="m 139.5,33 9,0 -1.5,10 -6,0" - id="path3591" - inkscape:connector-curvature="0" /><path - d="m 147.5,30 -2,0 0,-1 -3,0 0,1 -2,0 c -0.55,0 -1,0.48 -1,1 l 0,1 1,0 7,0 1,0 0,-1 c 0,-0.52 -0.45,-1 -1,-1" - id="path3593" - inkscape:connector-curvature="0" /><path - d="m 317,69 0,-5 -5,5 5,0 z" - id="path3595" - inkscape:connector-curvature="0" /><path - d="m 23,84 -8,-5 0,10" - id="path3597" - inkscape:connector-curvature="0" /><path - d="m 13,89 -3,0 0,-10 3,0 0,10 z" - id="path3599" - inkscape:connector-curvature="0" /><path - d="m 47,88 -3,0 0,-8 3,0 0,8 z" - id="path3601" - inkscape:connector-curvature="0" /><path - d="m 53,88 -3,0 0,-8 3,0 0,8 z" - id="path3603" - inkscape:connector-curvature="0" /><path - d="m 78.5,89 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" - id="path3605" - inkscape:connector-curvature="0" /><path - d="m 79,78 0,4 -3,0 4.5,4 4.5,-4 -3,0 0,-4 -3,0 z" - id="path3607" - inkscape:connector-curvature="0" /><path - d="m 110.5,89 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" - id="path3609" - inkscape:connector-curvature="0" /><path - d="m 112.5,78 -4.5,4 3,0 0,4 3,0 0,-4 3,0 -4.5,-4 z" - id="path3611" - inkscape:connector-curvature="0" /><path - d="m 142,86.5 c 0,1.1 0.9,2 2,2 1.1,0 2,-0.9 2,-2 0,-1.1 -0.9,-2 -2,-2 -1.1,0 -2,0.9 -2,2" - id="path3613" - inkscape:connector-curvature="0" /><path - d="m 137.25,87.03 c 2.55,-8.43 11.4,-8.73 13.94,0" - id="path3615" - inkscape:connector-curvature="0" - style="fill:none;stroke:#000000;stroke-width:2.5" /><path - d="m 151.68,89 -4.54,-2.76 6.68,-2.1" - id="path3617" - inkscape:connector-curvature="0" /><path - d="m 12,102 -7,-4 0,8" - id="path3619" - inkscape:connector-curvature="0" /><path - d="m 24,106 4,-7 -8,0" - id="path3621" - inkscape:connector-curvature="0" /><path - d="m 8,111 4,7 -8,0" - id="path3623" - inkscape:connector-curvature="0" /><path - d="m 54,105 0,7 -10,0 0,-7 z m 1,8 0,-11 -12,0 0,11 z" - id="path3625" - inkscape:connector-curvature="0" /><rect - height="3" - ry="0" - width="12" - x="73" - y="110" - id="rect3627" - style="opacity:0.81999984" /><rect - height="12" - width="14" - x="168.5" - y="78.5" - id="rect3629" - style="fill:none;stroke:#000000" /><path - d="m 180,84.5 -5,-3.25 0,6.5" - id="path3631" - inkscape:connector-curvature="0" /><rect - height="13" - width="1" - x="172" - y="78" - id="rect3633" /><rect - height="12" - width="14" - x="200.5" - y="78.5" - id="rect3635" - style="fill:none;stroke:#000000" /><path - d="m 207,84.5 5,-3.25 0,6.5" - id="path3637" - inkscape:connector-curvature="0" /><rect - height="13" - width="1" - x="204" - y="78" - id="rect3639" /><rect - height="12" - width="14" - x="264.5" - y="78.5" - id="rect3641" - style="fill:none;stroke:#000000" /><path - d="m 272,84.5 -5,-3.25 0,6.5" - id="path3643" - inkscape:connector-curvature="0" /><rect - height="13" - width="1" - x="274" - y="78" - id="rect3645" /><rect - height="12" - width="14" - x="296.5" - y="78.5" - id="rect3647" - style="fill:none;stroke:#000000" /><path - d="m 299,84.5 5,-3.25 0,6.5" - id="path3649" - inkscape:connector-curvature="0" /><rect - height="13" - width="1" - x="306" - y="78" - id="rect3651" /><path - d="m 118,96 c -3.87,0 -7,3.13 -7,7 0,3.87 3.13,7 7,7 3.87,0 7,-3.13 7,-7 0,-3.87 -3.13,-7 -7,-7" - id="path3653" - inkscape:connector-curvature="0" - style="fill:url(#m)" /><path - d="m 111.5,103 c 0,3.59 2.91,6.5 6.5,6.5 3.59,0 6.5,-2.91 6.5,-6.5 0,-3.59 -2.91,-6.5 -6.5,-6.5 -3.59,0 -6.5,2.91 -6.5,6.5" - id="path3655" - inkscape:connector-curvature="0" - style="fill:#f27d82" /><path - d="m 121.5,100.93 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3657" - inkscape:connector-curvature="0" - style="fill-opacity:0.36000001" /><path - d="m 121.5,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3659" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><path - d="m 138.5,100.93 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3661" - inkscape:connector-curvature="0" - style="fill-opacity:0.23999999" /><path - d="m 138.5,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3663" - inkscape:connector-curvature="0" - style="fill:#676767" /><path - d="m 103,110 c 3.87,0 7,-3.13 7,-7 0,-3.87 -3.13,-7 -7,-7 -3.87,0 -7,3.13 -7,7 0,3.87 3.13,7 7,7" - id="path3665" - inkscape:connector-curvature="0" - style="fill:url(#o)" /><path - d="m 103,96.5 c -3.59,0 -6.5,2.91 -6.5,6.5 0,3.59 2.91,6.5 6.5,6.5 3.59,0 6.5,-2.91 6.5,-6.5 0,-3.59 -2.91,-6.5 -6.5,-6.5" - id="path3667" - inkscape:connector-curvature="0" - style="fill:url(#n)" /><path - d="m 106.5,100.93 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3669" - inkscape:connector-curvature="0" - style="fill:#993c35" /><path - d="m 106.5,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3671" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><path - d="m 143,102.5 c 0,3.59 2.91,6.5 6.5,6.5 3.59,0 6.5,-2.91 6.5,-6.5 0,-3.59 -2.91,-6.5 -6.5,-6.5 -3.59,0 -6.5,2.91 -6.5,6.5" - id="path3673" - inkscape:connector-curvature="0" - style="fill:#bebebe" /><path - d="m 153,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3675" - inkscape:connector-curvature="0" - style="fill-opacity:0.37000002" /><path - d="M 153,99.93 152.07,99 149.5,101.57 146.93,99 146,99.93 l 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3677" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><path - d="m 160,102.5 c 0,3.59 2.91,6.5 6.5,6.5 3.59,0 6.5,-2.91 6.5,-6.5 0,-3.59 -2.91,-6.5 -6.5,-6.5 -3.59,0 -6.5,2.91 -6.5,6.5" - id="path3679" - inkscape:connector-curvature="0" - style="fill:#9f9f9f" /><path - d="m 170,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3681" - inkscape:connector-curvature="0" - style="fill-opacity:0.36000001" /><path - d="M 170,99.93 169.07,99 166.5,101.57 163.93,99 163,99.93 l 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3683" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><path - d="m 184.5,100.43 -0.93,-0.93 -2.57,2.57 -2.57,-2.57 -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3685" - inkscape:connector-curvature="0" - style="fill-opacity:0.23999999" /><path - d="M 184.5,99.93 183.57,99 181,101.57 178.43,99 l -0.93,0.93 2.57,2.57 -2.57,2.57 0.93,0.93 2.57,-2.57 2.57,2.57 0.93,-0.93 -2.57,-2.57 2.57,-2.57 z" - id="path3687" - inkscape:connector-curvature="0" - style="fill:#676767" /><path - d="m 131.65,116.21 -1.44,-2.03 -1.21,1.21 2.55,3.61 6.45,-7.67 -1.12,-1.33 z" - id="path3689" - inkscape:connector-curvature="0" /><path - d="m 195.75,97.75 3.5,3.25 -3.5,3.25" - id="path3691" - inkscape:connector-curvature="0" - style="fill:none;stroke:#367cf1;stroke-width:1.5" /><path - d="m 195.75,108.75 3.5,3.25 -3.5,3.25" - id="path3693" - inkscape:connector-curvature="0" - style="fill:none;stroke:#939393;stroke-width:1.5" /><path - d="m 209,101 c 0,0.55 0.45,1 1,1 0.55,0 1,-0.45 1,-1 0,-0.55 -0.45,-1 -1,-1 -0.55,0 -1,0.45 -1,1" - id="path3695" - inkscape:connector-curvature="0" - style="fill:#bababa" /><path - d="m 208.25,97.75 -3.5,3.25 3.5,3.25" - id="path3697" - inkscape:connector-curvature="0" - style="fill:none;stroke:#bababa;stroke-width:1.5" /><path - d="m 218,96 c -2.76,0 -5,2.24 -5,5 0,2.76 2.24,5 5,5 2.76,0 5,-2.24 5,-5 0,-2.76 -2.24,-5 -5,-5" - id="path3699" - inkscape:connector-curvature="0" - style="fill:url(#p)" /><path - d="m 213.36,101 c 0,2.56 2.08,4.64 4.64,4.64 2.56,0 4.64,-2.08 4.64,-4.64 0,-2.56 -2.08,-4.64 -4.64,-4.64 -2.56,0 -4.64,2.08 -4.64,4.64" - id="path3701" - inkscape:connector-curvature="0" - style="fill:#eb3941" /><path - d="m 216,99 4,4" - id="path3703" - inkscape:connector-curvature="0" - style="stroke:#ffffff" /><path - d="m 220,99 -4,4" - id="path3705" - inkscape:connector-curvature="0" - style="stroke:#ffffff" /><path - d="m 203,116 4,-8 4,8 z" - id="path3707" - inkscape:connector-curvature="0" - style="stroke:#c19600;stroke-width:2;stroke-linejoin:round" /><path - d="m 203,116 4,-8 4,8 z" - id="path3709" - inkscape:connector-curvature="0" - style="fill:#f4bd00;stroke:#f5bd00;stroke-width:1.5;stroke-linejoin:round" /><path - d="m 205.75,109.75 2.5,0 0,2.5 -0.5,1.75 -1.5,0 -0.5,-1.75 0,-2.5 m 0,5.25 2.5,0 0,1.25 -2.5,0" - id="path3711" - inkscape:connector-curvature="0" - style="fill:#ad8601" /><path - d="m 206,110 2,0 0,2.25 -0.5,1.75 -1,0 -0.5,-1.75 0,-2.25 m 0,5 2,0 0,1 -2,0" - id="path3713" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><path - d="m 229,106 c -2.76,0 -5,-2.24 -5,-5 0,-2.76 2.24,-5 5,-5 2.76,0 5,2.24 5,5 0,2.76 -2.24,5 -5,5 z" - id="path3715" - inkscape:connector-curvature="0" - style="fill:url(#r)" /><path - d="m 233.5,101 c 0,2.49 -2.01,4.5 -4.5,4.5 -2.49,0 -4.5,-2.01 -4.5,-4.5 0,-2.49 2.01,-4.5 4.5,-4.5 2.49,0 4.5,2.01 4.5,4.5 z" - id="path3717" - inkscape:connector-curvature="0" - style="fill:#dd0000" /><path - d="m 229.03,96.53 c 1.97,0 3.56,1.02 3.56,2.28 0,1.26 -1.59,2.28 -3.56,2.28 -1.97,0 -3.56,-1.02 -3.56,-2.28 0,-1.26 1.59,-2.28 3.56,-2.28 z" - id="path3719" - inkscape:connector-curvature="0" - style="fill:url(#s)" /><path - d="m 229.03,105.47 c 1.66,0 3,-0.71 3,-1.59 0,-0.88 -1.34,-1.59 -3,-1.59 -1.66,0 -3,0.71 -3,1.59 0,0.88 1.34,1.59 3,1.59 z" - id="path3721" - inkscape:connector-curvature="0" - style="fill:url(#t)" /><path - d="m 239.95,106 c -2.76,-0.03 -4.98,-2.29 -4.95,-5.05 0.03,-2.76 2.29,-4.98 5.05,-4.95 2.76,0.03 4.98,2.29 4.95,5.05 -0.03,2.76 -2.29,4.98 -5.05,4.95 z" - id="path3723" - inkscape:connector-curvature="0" - style="fill:url(#u)" /><path - d="m 244.5,101.05 c -0.03,2.49 -2.06,4.48 -4.55,4.45 -2.49,-0.03 -4.48,-2.06 -4.45,-4.55 0.03,-2.49 2.06,-4.48 4.55,-4.45 2.49,0.03 4.48,2.06 4.45,4.55 z" - id="path3725" - inkscape:connector-curvature="0" - style="fill:#00be00" /><path - d="m 240.08,96.53 c 1.97,0.02 3.55,1.06 3.54,2.32 -0.01,1.26 -1.62,2.26 -3.59,2.24 -1.97,-0.02 -3.55,-1.06 -3.54,-2.32 0.01,-1.26 1.62,-2.26 3.59,-2.24 z" - id="path3727" - inkscape:connector-curvature="0" - style="fill:url(#v)" /><path - d="m 239.98,105.41 c 1.66,0.02 3.01,-0.68 3.02,-1.56 0.01,-0.88 -1.33,-1.61 -2.98,-1.63 -1.66,-0.02 -3.01,0.68 -3.02,1.56 -0.01,0.88 1.33,1.61 2.98,1.63 z" - id="path3729" - inkscape:connector-curvature="0" - style="fill:url(#w)" /><path - d="m 251,106 c -2.76,0 -5,-2.24 -5,-5 0,-2.76 2.24,-5 5,-5 2.76,0 5,2.24 5,5 0,2.76 -2.24,5 -5,5 z" - id="path3731" - inkscape:connector-curvature="0" - style="fill:#e5a600" /><path - d="m 255.5,101 c 0,2.49 -2.01,4.5 -4.5,4.5 -2.49,0 -4.5,-2.01 -4.5,-4.5 0,-2.49 2.01,-4.5 4.5,-4.5 2.49,0 4.5,2.01 4.5,4.5 z" - id="path3733" - inkscape:connector-curvature="0" - style="fill:#ffbd00" /><path - d="m 251.03,96.53 c 1.97,0 3.56,1.02 3.56,2.28 0,1.26 -1.59,2.28 -3.56,2.28 -1.97,0 -3.56,-1.02 -3.56,-2.28 0,-1.26 1.59,-2.28 3.56,-2.28 z" - id="path3735" - inkscape:connector-curvature="0" - style="fill:url(#x)" /><path - d="m 250.99,105.42 c 1.66,0 3,-0.71 3,-1.59 0,-0.88 -1.34,-1.59 -3,-1.59 -1.66,0 -3,0.71 -3,1.59 0,0.88 1.34,1.59 3,1.59 z" - id="path3737" - inkscape:connector-curvature="0" - style="fill:url(#y)" /><rect - height="12" - rx="0.25" - ry="0.25" - style="fill-opacity:0.38000016;stroke:#000000;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:0" - width="8" - x="268" - y="102" - id="rect3739" /><path - d="m 218,107 c -2.76,0 -5,2.24 -5,5 0,2.76 2.24,5 5,5 2.76,0 5,-2.24 5,-5 0,-2.76 -2.24,-5 -5,-5" - id="path3741" - inkscape:connector-curvature="0" - style="fill:url(#q)" /><path - d="m 213.36,112 c 0,2.56 2.08,4.64 4.64,4.64 2.56,0 4.64,-2.08 4.64,-4.64 0,-2.56 -2.08,-4.64 -4.64,-4.64 -2.56,0 -4.64,2.08 -4.64,4.64" - id="path3743" - inkscape:connector-curvature="0" - style="fill:#2a53cd" /><path - d="m 216.93,109.14 c -0.03,-0.53 0.55,-0.97 1.06,-0.83 0.5,0.12 0.79,0.73 0.56,1.18 -0.2,0.44 -0.79,0.61 -1.2,0.36 -0.26,-0.14 -0.42,-0.42 -0.42,-0.71 z m 1.7,5.46 c 0.22,0 0.45,0 0.67,0 0,0.18 0,0.35 0,0.53 -0.96,0 -1.93,0 -2.89,0 0,-0.18 0,-0.35 0,-0.53 0.22,0 0.44,0 0.66,0 0,-1.2 0,-2.41 0,-3.61 -0.22,0 -0.44,0 -0.66,0 0,-0.18 0,-0.35 0,-0.53 0.74,0 1.48,0 2.22,0 0,1.38 0,2.76 0,4.14 z" - id="path3745" - inkscape:connector-curvature="0" - style="fill:#ffffff" /><rect - height="8" - rx="0.25" - ry="0.25" - style="fill-opacity:0.38000016;stroke:#000000;stroke-width:2;stroke-linejoin:round;stroke-miterlimit:0" - width="12" - x="298" - y="104" - id="rect3747" /><path - d="m 12.25,125 c -0.68,0 -1.25,0.57 -1.25,1.25 l 0,9.69 2,-3.31 0,-5.63 3.38,0 1.19,-2 -5.31,0 z m 8.75,3.06 -2,3.31 0,5.63 -3.38,0 -1.19,2 5.31,0 c 0.68,0 1.25,-0.57 1.25,-1.25 l 0,-9.69 z" - id="path3749" - inkscape:connector-curvature="0" - style="stroke-width:2" /><path - d="m 10.75,140.75 10.5,-17.5" - id="path3751" - inkscape:connector-curvature="0" - style="fill:none;stroke:#000000;stroke-width:2" /><path - d="m 46,128 7,0 0,8 -7,0 0,-8 z m 9,-2 -14,0 0,12 14,0 0,-12 z" - id="path3753" - inkscape:connector-curvature="0" /><rect - height="6" - width="3" - x="73" - y="129" - id="rect3755" /><rect - height="6" - width="3" - x="77" - y="129" - id="rect3757" /><rect - height="6" - width="3" - x="81" - y="129" - id="rect3759" /><rect - height="10" - rx="1" - ry="1" - width="15" - x="71" - y="127" - id="rect3761" - style="fill:none;stroke:#000000;stroke-width:2;stroke-linejoin:round" /><path - d="m 86,129 c 3,0 3,0 3,3 0,3 0,3 -3,3" - id="path3763" - inkscape:connector-curvature="0" - style="fill:none;stroke:#000000;stroke-width:2" /><path - d="m 227.25,107 c -0.7,0 -1.25,0.5 -1.25,1.25 l 0,7.5 c 0,0.7 0.5,1.25 1.25,1.25 l 3.5,0 c 0.7,0 1.25,-0.5 1.25,-1.25 l 0,-7.5 c 0,-0.7 -0.5,-1.25 -1.25,-1.25 l -3.5,0 z m -0.25,1 4,0 0,7 -4,0 0,-7 z m 2,7.25 c 0.4,0 0.75,0.3 0.75,0.75 0,0.4 -0.3,0.75 -0.75,0.75 -0.4,0 -0.75,-0.3 -0.75,-0.75 0,-0.4 0.3,-0.75 0.75,-0.75 z" - id="path3765" - inkscape:connector-curvature="0" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - d="m 110.49999,127.4989 -1,0 2,-2 2,2 -1,0 c 0,0 -0.0345,4.6379 -0.0345,4.0345 l 4.03448,-0.034 0,-1 2,2 -2,2 0,-1 -4.03449,-0.034 0.0345,4.0346 1,0 -2,2 -2,-2 1,0 0.0345,-4.0346 -4.03448,0.034 0,1 -2,-2 2,-2 0,1 4.03448,0.034 z" - id="path3550" - inkscape:connector-curvature="0" - sodipodi:nodetypes="ccccccccccccccccccccccccc" - inkscape:export-filename="/usr/local/google/home/caseq/aa.png" - inkscape:export-xdpi="90" - inkscape:export-ydpi="90" /><path - sodipodi:type="arc" - style="fill:#000000;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:2.77777767;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - id="path3635" - sodipodi:cx="50" - sodipodi:cy="14" - sodipodi:rx="3" - sodipodi:ry="3" - d="m 53,14 a 3,3 0 1 1 -6,0 3,3 0 1 1 6,0 z" - transform="matrix(0.36,0,0,0.36,125.49998,127.46)" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - d="m 143.48385,129.5 2.54033,-2 -1.5242,0 0,-2 -2,0 -0.0161,2 -1.54033,0 z" - id="path4411-7" - inkscape:connector-curvature="0" - sodipodi:nodetypes="cccccccc" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - d="m 146.49998,132.5 2,2.5 0,-1.4998 2,-1e-4 0,-2.0002 -2,1e-4 0,-1.5 z" - id="path4411-7-7" - inkscape:connector-curvature="0" - sodipodi:nodetypes="cccccccc" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - d="m 143.49998,135.5 -2.5,2 1.5,0 0,2 2,0 0,-2 1.5,0 z" - id="path4411-7-7-3" - inkscape:connector-curvature="0" - sodipodi:nodetypes="cccccccc" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - d="m 140.49998,132.5 -2,-2.5 0,1.4999 -2,0 0,2.0002 2,-3e-4 0,1.5002 z" - id="path4411-7-7-3-5" - inkscape:connector-curvature="0" - sodipodi:nodetypes="cccccccc" /><path - style="fill:#000000;fill-opacity:1;stroke:none" - inkscape:connector-curvature="0" - d="m 169.19838,130.1487 c 0.0551,-0.1128 0.15339,-0.22 0.2313,-0.3064 0.54584,-0.6074 1.38893,-1.1974 3.00529,-1.5702 l 0,-2.1715 c -1.29309,0.2523 -2.31462,0.6686 -3.304,1.5597 -0.33248,0.3832 -0.50543,0.6247 -0.514,1.1665 -0.007,0.443 0.17732,0.8863 0.58141,1.3219 z m 6.63093,3.1292 c -1.61635,-0.024 -3.10906,-0.2558 -4.62277,-0.764 -2.69382,-0.9037 -3.02857,-2.1697 -3.17032,-2.5931 -0.0958,1.6897 0.028,2.9573 0.0346,3.0648 0.092,1.5004 1.29179,2.6742 2.15751,3.1151 1.7796,0.9067 3.66137,1.3756 5.60099,1.4151 l 0,2.4842 3.25098,-4.6483 -3.25098,-4.6486 0,2.5748 z m -0.32327,-5.2959 c 1.61636,0.01 3.45286,0.3384 5.2454,1.1131 0.73528,0.3177 1.32751,0.7904 1.80821,1.3048 0.33152,-0.1523 0.80672,-0.7308 0.74126,-1.2923 -0.16163,-1.3879 -1.62637,-2.1912 -1.88014,-2.3045 -1.7704,-0.7908 -3.9751,-1.0356 -5.91473,-1.0589 l 0,-1.7442 -2.01317,2.8787 2.01317,2.8783 0,-1.775 z m 8.42363,1.8658 c -0.0824,0.6478 -0.82676,2.0349 -3.89784,2.9138 l 0,4.5247 c 1.13145,-0.3677 2.46495,-1.2956 3.17873,-2.1988 0.2911,-0.3677 0.46906,-0.875 0.57736,-1.242 0.33216,-1.1229 0.18604,-3.3136 0.14175,-3.9977 z" - id="Rotate" - sketch:type="MSShapeGroup" /><path - inkscape:connector-curvature="0" - d="M 233,88.08946 V 91 h 2.91054 L 244.1944,82.71614 241.28385,79.8056 233,88.08946 z m 13.76911,-7.9387 c 0.30785,-0.30785 0.30785,-0.79294 0,-1.10079 l -1.81908,-1.81909 c -0.30785,-0.30784 -0.79295,-0.30784 -1.10078,0 l -1.52058,1.52991 2.91054,2.91054 1.5299,-1.52057 z" - id="path4" /><path - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:1.99999976;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4.19999981;stroke-opacity:1;stroke-dasharray:none" - d="m 107,33 8,0 0,6 -8,0 z" - id="path3963" - inkscape:connector-curvature="0" - sodipodi:nodetypes="ccccc" /><g - id="g3234"><path - style="fill:url(#linearGradient3239)" - inkscape:connector-curvature="0" - id="path3699-5" - d="m 250,107 c -2.76,0 -5,2.24 -5,5 0,2.76 2.24,5 5,5 2.76,0 5,-2.24 5,-5 0,-2.76 -2.24,-5 -5,-5" /><path - id="path3701-0" - d="m 245.36,112 c 0,2.56 2.08,4.64 4.64,4.64 2.56,0 4.64,-2.08 4.64,-4.64 0,-2.56 -2.08,-4.64 -4.64,-4.64 -2.56,0 -4.64,2.08 -4.64,4.64" - inkscape:connector-curvature="0" - style="fill:#f4979b;fill-opacity:1" /><path - style="fill:#ffffff;fill-opacity:1" - id="path3689-2" - d="m 248.64912,113.55532 -1.12,-1.57887 -0.94113,0.9411 1.98332,2.80777 5.01668,-5.96555 -0.8711,-1.03445 z" - inkscape:connector-curvature="0" /></g><g - id="Animations-Toggle" - transform="translate(199.99866,124.00442)"><path - d="M 6,2 12,8 6,14 0,8" - id="path3453" - inkscape:connector-curvature="0" /><path - d="M 8.5,12.5 10,14 16,8 10,2 8.5,3.5 13,8 8.5,12.5 z" - id="path3455" - inkscape:connector-curvature="0" - style="opacity:0.5" /></g><path - inkscape:connector-curvature="0" - d="m 240.76701,127 -1.53402,0 0,4.23299 -4.23299,0 0,1.53402 4.23299,0 0,4.23299 1.53402,0 0,-4.23299 4.23299,0 0,-1.53402 -4.23299,0 0,-4.23299 z" - id="path3509-0" /><path - style="fill:none" - inkscape:connector-curvature="0" - d="m 228.56466,156.76923 h 48 v 48 h -48 z" - id="path3712" /><path - inkscape:connector-curvature="0" - d="m 310.7696,127.0439 -1.81599,-1.81636 c -0.30331,-0.30338 -0.79716,-0.30338 -1.10048,0 l -2.43038,2.43089 -1.48935,-1.49354 -1.10047,1.1007 1.10437,1.1046 L 297,135.30504 V 139 h 3.69419 l 6.9373,-6.93874 1.10048,1.1046 1.10048,-1.10072 -1.49324,-1.49353 2.43039,-2.43089 c 0.3072,-0.30338 0.3072,-0.79345 0,-1.09682 z m -10.72092,10.40033 -1.49324,-1.49354 6.27235,-6.27365 1.49323,1.49354 -6.27234,6.27365 z" - id="path3714" /><g - id="Page-1" - sketch:type="MSPage" - transform="translate(328,28)" - style="fill:none;stroke:none"><g - id="Replay" - sketch:type="MSArtboardGroup" - style="fill:#000000"><path - d="M 2.5858334,13.890843 C 0.99596482,12.428959 0,10.331325 0,8 0,3.58 3.58,0 8,0 c 4.42,0 8,3.58 8,8 0,4.42 -3.58,8 -8,8 l 0,-1.6 0,0 c 3.528,0 6.4,-2.872 6.4,-6.4 C 14.4,4.472 11.528,1.6 8,1.6 4.472,1.6 1.6,4.472 1.6,8 c 0,1.9357374 0.8646098,3.673987 2.2277938,4.848714 L 5.520254,11.428571 5.5353988,15.891545 1.1428571,15.101644 2.5858334,13.890843 z M 6.5066666,11.428571 11.078095,8 6.5066666,4.5714286 l 0,6.8571424 z" - id="Shape" - sketch:type="MSShapeGroup" - inkscape:connector-curvature="0" /></g></g><path - d="m 276,127 1,-1 0,-1 -8,0 0,1 1,1 0,5 -2,1 0,2 4,0 -0.0447,3.9835 L 273,140 l 1,-1 0,-4 4,0 0,-2 -2,-1 0,-5 z" - id="path3214" - inkscape:connector-curvature="0" - style="fill:#010101" - sodipodi:nodetypes="ccccccccccccccccccc" /><path - d="M 0,24 H 24 V 48 H 0 z" - id="path3216" - inkscape:connector-curvature="0" - style="fill:none" /><path - d="m 328,60 c 0,4.42 3.58,8 8,8 4.42,0 8,-3.58 8,-8 0,-4.42 -3.58,-8 -8,-8 -4.42,0 -8,3.58 -8,8 z m 8,6.4 c -3.528,0 -6.4,-2.872 -6.4,-6.4 0,-3.528 2.872,-6.4 6.4,-6.4 3.528,0 6.4,2.872 6.4,6.4 0,3.528 -2.872,6.4 -6.4,6.4 z M 334.50667,63.428571 339.0781,60 l -4.57143,-3.428571 0,6.857142 z" - id="Icon" - sketch:type="MSShapeGroup" - inkscape:connector-curvature="0" /><path - d="m 333.6,15.2 1.6,0 0,-6.4 -1.6,0 0,6.4 0,0 z M 336,4 c -4.42,0 -8,3.58 -8,8 0,4.42 3.58,8 8,8 4.42,0 8,-3.58 8,-8 0,-4.42 -3.58,-8 -8,-8 l 0,0 z m 0,14.4 c -3.528,0 -6.4,-2.872 -6.4,-6.4 0,-3.528 2.872,-6.4 6.4,-6.4 3.528,0 6.4,2.872 6.4,6.4 0,3.528 -2.872,6.4 -6.4,6.4 l 0,0 z m 0.8,-3.2 1.6,0 0,-6.4 -1.6,0 0,6.4 0,0 z" - id="path3448" - sketch:type="MSShapeGroup" - inkscape:connector-curvature="0" /><path - inkscape:connector-curvature="0" - id="path3871" - d="m 115,36 4,-4 0,8" /><path - d="M 0,24 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z m 0,0 H 24 V 48 H 0 z" - id="path3510" - inkscape:connector-curvature="0" - style="fill:none" /><path - d="m 208.00365,8.385948 c 2.00818,0 3.63802,1.629832 3.63802,3.638017 0,0.472943 -0.0946,0.916781 -0.26194,1.331515 l 2.1246,2.124602 c 1.09868,-0.91678 1.96453,-2.102774 2.49568,-3.456117 -1.25875,-3.194178 -4.36562,-5.457025 -8.00364,-5.457025 -1.01864,0 -1.99363,0.181901 -2.89586,0.509322 l 1.57162,1.571624 c 0.41474,-0.167349 0.85858,-0.261938 1.33152,-0.261938 z m -7.27604,-1.986357 1.65894,1.658936 0.3347,0.334697 c -1.20782,0.938609 -2.15371,2.190087 -2.72124,3.630741 1.25875,3.19418 4.36562,5.457026 8.00364,5.457026 1.12778,0 2.20464,-0.218281 3.1869,-0.611187 l 0.30559,0.305594 2.13188,2.124602 0.92406,-0.924056 -12.90041,-12.90041 -0.92406,0.924057 z m 4.02365,4.023647 1.12779,1.127785 c -0.0364,0.152797 -0.0582,0.31287 -0.0582,0.472942 0,1.207822 0.97499,2.182811 2.18281,2.182811 0.16007,0 0.32014,-0.02183 0.47294,-0.05821 l 1.12779,1.127785 c -0.4875,0.240109 -1.02593,0.38563 -1.60073,0.38563 -2.00819,0 -3.63802,-1.629832 -3.63802,-3.638018 0,-0.574806 0.14552,-1.113233 0.38563,-1.600727 z m 3.13597,-0.567531 2.29195,2.291951 0.0145,-0.116417 c 0,-1.207821 -0.97498,-2.18281 -2.18281,-2.18281 l -0.12369,0.0073 z" - id="path3512" - inkscape:connector-curvature="0" /><path - d="M 0,24 H 24 V 48 H 0 z" - id="path3223" - inkscape:connector-curvature="0" - style="fill:none" /><path - d="m 111.99999,6.690909 c -3.63635,0 -6.74181,2.261818 -7.99999,5.454546 1.25818,3.192727 4.36364,5.454545 7.99999,5.454545 3.63636,0 6.74182,-2.261818 8,-5.454545 -1.25818,-3.192728 -4.36364,-5.454546 -8,-5.454546 z m 0,9.090909 c -2.00726,0 -3.63635,-1.629091 -3.63635,-3.636363 0,-2.007273 1.62909,-3.636364 3.63635,-3.636364 2.00727,0 3.63636,1.629091 3.63636,3.636364 0,2.007272 -1.62909,3.636363 -3.63636,3.636363 z m 0,-5.818182 c -1.20726,0 -2.18181,0.974546 -2.18181,2.181819 0,1.207272 0.97455,2.181818 2.18181,2.181818 1.20727,0 2.18182,-0.974546 2.18182,-2.181818 0,-1.207273 -0.97455,-2.181819 -2.18182,-2.181819 z" - id="path3225" - inkscape:connector-curvature="0" /><path - d="M 0,24 H 24 V 48 H 0 z" - id="path3224" - inkscape:connector-curvature="0" - style="fill:none" /><path - d="m 338.33036,76.32322 h -4.57143 v 1.52381 h 4.57143 v -1.52381 z m -3.04762,9.90477 h 1.52381 v -4.57143 h -1.52381 v 4.57143 z m 6.11809,-5.03619 1.08191,-1.08191 c -0.32762,-0.38857 -0.68572,-0.75428 -1.07429,-1.07428 l -1.0819,1.0819 c -1.18095,-0.94476 -2.66667,-1.50857 -4.28191,-1.50857 -3.78667,0 -6.85714,3.07047 -6.85714,6.85714 0,3.78667 3.06285,6.85714 6.85714,6.85714 3.79429,0 6.85715,-3.07047 6.85715,-6.85714 0,-1.61524 -0.56382,-3.10095 -1.50096,-4.27428 z m -5.35619,9.60761 c -2.94857,0 -5.33333,-2.38476 -5.33333,-5.33333 0,-2.94857 2.38476,-5.33333 5.33333,-5.33333 2.94857,0 5.33333,2.38476 5.33333,5.33333 0,2.94857 -2.38476,5.33333 -5.33333,5.33333 z" - id="path3226" - inkscape:connector-curvature="0" /><path - d="m 334,114 10,0 0,-5 -10,0 z m -6,0 5,0 0,-12 -5,0 z m 6,-12 0,6 10,0 0,-6 z" - id="path3227" - inkscape:connector-curvature="0" - sodipodi:nodetypes="ccccccccccccccc" /><path - d="M 0,24 H 24 V 48 H 0 z" - id="path3229" - inkscape:connector-curvature="0" - style="fill:none" /><path - sodipodi:type="arc" - style="fill:#000000;fill-opacity:1;fill-rule:evenodd;stroke:none" - id="path3518" - sodipodi:cx="208.86168" - sodipodi:cy="31.304852" - sodipodi:rx="1.0606602" - sodipodi:ry="1.0606602" - d="m 209.92234,31.304852 a 1.0606602,1.0606602 0 1 1 -2.12132,0 1.0606602,1.0606602 0 1 1 2.12132,0 z" - transform="matrix(1.4142135,0,0,1.4142135,-86.874996,-12.771743)" /><path - transform="matrix(1.4142135,0,0,1.4142135,-86.874986,-7.771742)" - d="m 209.92234,31.304852 a 1.0606602,1.0606602 0 1 1 -2.12132,0 1.0606602,1.0606602 0 1 1 2.12132,0 z" - sodipodi:ry="1.0606602" - sodipodi:rx="1.0606602" - sodipodi:cy="31.304852" - sodipodi:cx="208.86168" - id="path4288" - style="fill:#000000;fill-opacity:1;fill-rule:evenodd;stroke:none" - sodipodi:type="arc" /><path - sodipodi:type="arc" - style="fill:#000000;fill-opacity:1;fill-rule:evenodd;stroke:none" - id="path4290" - sodipodi:cx="208.86168" - sodipodi:cy="31.304852" - sodipodi:rx="1.0606602" - sodipodi:ry="1.0606602" - d="m 209.92234,31.304852 a 1.0606602,1.0606602 0 1 1 -2.12132,0 1.0606602,1.0606602 0 1 1 2.12132,0 z" - transform="matrix(1.4142135,0,0,1.4142135,-86.874986,-2.771743)" /><path - inkscape:connector-curvature="0" - id="path3246" - d="M 86,38 83.277778,35 72,35 l 0,6 11.277778,0" /><rect - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:0.94679004;stroke-miterlimit:3.5;stroke-opacity:1;stroke-dasharray:none" - id="rect4034" - width="4.0532098" - height="0.053209968" - x="85.473381" - y="32.473396" /><rect - y="30.473396" - x="87.473381" - height="4.0532098" - width="0.053209968" - id="rect4036" - style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:0.94679004;stroke-miterlimit:3.5;stroke-opacity:1;stroke-dasharray:none" /><path - inkscape:connector-curvature="0" - d="m 335,138 -3.66666,0 C 330.59667,138 330,137.40333 330,136.66667 l 0,-9.33334 C 330,126.59667 330.59667,126 331.33334,126 l 9.33333,0 c 0.73667,0 1.33333,0.59667 1.33333,1.33333 l 0,3.66667 -2,-0.6 0,-2.4 -8,-3.4e-4 0,8.00017 2.22223,3.4e-4 z m 5.25657,-2.70598 2.40637,-2.40638 L 334,130 l 2.88765,8.66294 2.40637,-2.40638 L 343.03746,140 344,139.03745 z" - id="inspect_element" - sketch:type="MSShapeGroup" - sodipodi:nodetypes="cssssssccccccccccccccc" /><path - inkscape:connector-curvature="0" - d="M 178.71429,5.0064 172.28571,5 C 171.57857,5 171,5.572727 171,6.272727 l 0,11.454546 c 0,0.7 0.57857,1.272727 1.28571,1.272727 l 6.42858,0 C 179.42143,19 180,18.427273 180,17.727273 l 0,-11.454546 c 0,-0.7 -0.57857,-1.266363 -1.28571,-1.266363 l 0,0 z m 0,11.448182 -6.42858,0 0,-8.909091 6.42858,0 0,8.909091 0,0 z" - id="device-mode" - sketch:type="MSShapeGroup" /><path - d="m 23,157 -2,0 0,2 2,0 z" - id="path3288" /><path - d="m 23,161 -2,0 0,2 c 1,0 2,-1 2,-2 z" - id="path3286" /><path - d="m 23,153 -2,0 0,2 2,0 z" - id="path3284" /><path - d="m 21,149 0,2 2,0 c 0,-1 -1,-2 -2,-2 z" - id="path3282" /><path - d="m 11,163 4,0 0,-6 -6,0 0,4 c 0,1.1 0.9,2 2,2 z" - id="path3280" - inkscape:connector-curvature="0" - sodipodi:nodetypes="scccss" /><path - d="m 11,153 -2,0 0,2 2,0 z" - id="path3278" /><path - d="m 19,149 -2,0 0,2 2,0 z" - id="path3276" /><path - d="m 19,161 -2,0 0,2 2,0 z" - id="path3274" /><path - d="m 11,149 c -1,0 -2,1 -2,2 l 2,0 z" - id="path3272" /><path - d="m 15,149 -2,0 0,2 2,0 z" - id="path3231" /><g - id="g4042"><path - transform="matrix(1.1939089,0,0,1.1367841,-44.307792,-20.636427)" - d="m 80.87534,155.38257 a 4.3973203,4.6182914 0 1 1 -8.794641,0 4.3973203,4.6182914 0 1 1 8.794641,0 z" - sodipodi:ry="4.6182914" - sodipodi:rx="4.3973203" - sodipodi:cy="155.38257" - sodipodi:cx="76.47802" - id="path4006" - style="fill:none;stroke:#000000;stroke-width:1.2875576;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" - sodipodi:type="arc" /><rect - ry="0" - rx="0.38569456" - y="155" - x="44" - height="2" - width="6" - id="rect4010" - style="fill:#000000;fill-opacity:1;stroke:none" /></g><path - d="M 76.94,152 76,152.94 79.05334,156 76,159.06 l 0.94,0.94 4,-4 z" - id="path3240" - inkscape:connector-curvature="0" /><path - d="M0 0h24v24H0z" - fill="none" - id="path3242" /><path - d="M 80.94,152 80,152.94 83.05334,156 80,159.06 l 0.94,0.94 4,-4 z" - id="path3240-4" - inkscape:connector-curvature="0" /></svg> \ No newline at end of file diff --git a/1-js/03-code-quality/02-coding-style/1-style-errors/solution.md b/1-js/03-code-quality/02-coding-style/1-style-errors/solution.md index 764e36c63..4facc8b29 100644 --- a/1-js/03-code-quality/02-coding-style/1-style-errors/solution.md +++ b/1-js/03-code-quality/02-coding-style/1-style-errors/solution.md @@ -12,7 +12,7 @@ function pow(x,n) // <- no space between arguments let x=prompt("x?",''), n=prompt("n?",'') // <-- technically possible, // but better make it 2 lines, also there's no spaces and missing ; -if (n<0) // <- no spaces inside (n < 0), and should be extra line above it +if (n<=0) // <- no spaces inside (n <= 0), and should be extra line above it { // <- figure bracket on a separate line // below - long lines can be split into multiple lines for improved readability alert(`Power ${n} is not supported, please enter an integer number greater than zero`); @@ -39,7 +39,7 @@ function pow(x, n) { let x = prompt("x?", ""); let n = prompt("n?", ""); -if (n < 0) { +if (n <= 0) { alert(`Power ${n} is not supported, please enter an integer number greater than zero`); } else { diff --git a/1-js/03-code-quality/02-coding-style/article.md b/1-js/03-code-quality/02-coding-style/article.md index d265da434..904f0a939 100644 --- a/1-js/03-code-quality/02-coding-style/article.md +++ b/1-js/03-code-quality/02-coding-style/article.md @@ -2,11 +2,11 @@ Our code must be as clean and easy to read as possible. -That is actually the art of programming -- to take a complex task and code it in a way that is both correct and human-readable. +That is actually the art of programming -- to take a complex task and code it in a way that is both correct and human-readable. A good code style greatly assists in that. ## Syntax -Here is a cheatsheet with some suggested rules (see below for more details): +Here is a cheat sheet with some suggested rules (see below for more details):  <!-- @@ -26,7 +26,7 @@ let n = prompt("n?", ""); if (n < 0) { alert(`Power ${n} is not supported, - please enter an integer number, greater than 0`); + please enter a non-negative integer number`); } else { alert( pow(x, n) ); } @@ -36,7 +36,7 @@ if (n < 0) { Now let's discuss the rules and reasons for them in detail. -```warn header="Irony Detected" +```warn header="There are no \"you must\" rules" Nothing is set in stone here. These are style preferences, not religious dogmas. ``` @@ -52,33 +52,57 @@ if (condition) { } ``` -A single-line construct is an important edge case. Should we use brackets at all? If yes, then where? +A single-line construct, such as `if (condition) doSomething()`, is an important edge case. Should we use braces at all? Here are the annotated variants so you can judge their readability for yourself: -<!-- -```js no-beautify -if (n < 0) {alert(`Power ${n} is not supported`);} +1. 😠 Beginners sometimes do that. Bad! Curly braces are not needed: + ```js + if (n < 0) *!*{*/!*alert(`Power ${n} is not supported`);*!*}*/!* + ``` +2. 😠 Split to a separate line without braces. Never do that, easy to make an error when adding new lines: + ```js + if (n < 0) + alert(`Power ${n} is not supported`); + ``` +3. 😏 One line without braces - acceptable, if it's short: + ```js + if (n < 0) alert(`Power ${n} is not supported`); + ``` +4. 😃 The best variant: + ```js + if (n < 0) { + alert(`Power ${n} is not supported`); + } + ``` -if (n < 0) alert(`Power ${n} is not supported`); +For a very brief code, one line is allowed, e.g. `if (cond) return null`. But a code block (the last variant) is usually more readable. -if (n < 0) - alert(`Power ${n} is not supported`); +### Line Length -if (n < 0) { - alert(`Power ${n} is not supported`); -} -``` ---> - +No one likes to read a long horizontal line of code. It's best practice to split them. -In summary: -- For very short code, one line is acceptable. For example: `if (cond) return null`. -- But a separate line for each statement in brackets is usually easier to read. +For example: +```js +// backtick quotes ` allow to split the string into multiple lines +let str = ` + ECMA International's TC39 is a group of JavaScript developers, + implementers, academics, and more, collaborating with the community + to maintain and evolve the definition of JavaScript. +`; +``` -### Line Length +And, for `if` statements: -No one likes to read a long horizontal line of code. It's best practice to split them up and limit the length of your lines. +```js +if ( + id === 123 && + moonPhase === 'Waning Gibbous' && + zodiacSign === 'Libra' +) { + letTheSorceryBegin(); +} +``` The maximum line length should be agreed upon at the team-level. It's usually 80 or 120 characters. @@ -88,11 +112,11 @@ There are two types of indents: - **Horizontal indents: 2 or 4 spaces.** - A horizontal indentation is made using either 2 or 4 spaces or the "Tab" symbol. Which one to choose is an old holy war. Spaces are more common nowadays. + A horizontal indentation is made using either 2 or 4 spaces or the horizontal tab symbol (key `key:Tab`). Which one to choose is an old holy war. Spaces are more common nowadays. - One advantage of spaces over tabs is that spaces allow more flexible configurations of indents than the "Tab" symbol. + One advantage of spaces over tabs is that spaces allow more flexible configurations of indents than the tab symbol. - For instance, we can align the arguments with the opening bracket, like this: + For instance, we can align the parameters with the opening bracket, like this: ```js no-beautify show(parameters, @@ -127,15 +151,15 @@ There are two types of indents: A semicolon should be present after each statement, even if it could possibly be skipped. -There are languages where a semicolon is truly optional and it is rarely used. In JavaScript, though, there are cases where a line break is not interpreted as a semicolon, leaving the code vulnerable to errors. +There are languages where a semicolon is truly optional and it is rarely used. In JavaScript, though, there are cases where a line break is not interpreted as a semicolon, leaving the code vulnerable to errors. See more about that in the chapter <info:structure#semicolon>. -As you become more mature as a programmer, you may choose a no-semicolon style like [StandardJS](https://standardjs.com/). Until then, it's best to use semicolons to avoid possible pitfalls. +If you're an experienced JavaScript programmer, you may choose a no-semicolon code style like [StandardJS](https://standardjs.com/). Otherwise, it's best to use semicolons to avoid possible pitfalls. The majority of developers put semicolons. ### Nesting Levels Try to avoid nesting code too many levels deep. -Sometimes it's a good idea to use the ["continue"](info:while-for#continue) directive in a loop to avoid extra nesting. +For example, in the loop, it's sometimes a good idea to use the [`continue`](info:while-for#continue) directive to avoid extra nesting. For example, instead of adding a nested `if` conditional like this: @@ -197,13 +221,13 @@ function pow(x, n) { } ``` -The second one is more readable because the "edge case" of `n < 0` is handled early on. Once the check is done we can move on to the "main" code flow without the need for additional nesting. +The second one is more readable because the "special case" of `n < 0` is handled early on. Once the check is done we can move on to the "main" code flow without the need for additional nesting. ## Function Placement If you are writing several "helper" functions and the code that uses them, there are three ways to organize the functions. -1. Functions declared above the code that uses them: +1. Declare the functions *above* the code that uses them: ```js // *!*function declarations*/!* @@ -249,39 +273,39 @@ If you are writing several "helper" functions and the code that uses them, there Most of time, the second variant is preferred. -That's because when reading code, we first want to know *what it does*. If the code goes first, then it provides that information. Then, maybe we won't need to read the functions at all, especially if their names are descriptive of what they actually do. +That's because when reading code, we first want to know *what it does*. If the code goes first, then it becomes clear from the start. Then, maybe we won't need to read the functions at all, especially if their names are descriptive of what they actually do. ## Style Guides -A style guide contains general rules about "how to write" code, e.g. which quotes to use, how many spaces to indent, where to put line breaks, etc. A lot of minor things. +A style guide contains general rules about "how to write" code, e.g. which quotes to use, how many spaces to indent, the maximal line length, etc. A lot of minor things. When all members of a team use the same style guide, the code looks uniform, regardless of which team member wrote it. -Of course, a team can always write their own style guide. Most of the time though, there's no need to. There are many existing tried and true options to choose from, so adopting one of these is usually your best bet. +Of course, a team can always write their own style guide, but usually there's no need to. There are many existing guides to choose from. Some popular choices: -- [Google JavaScript Style Guide](https://google.github.io/styleguide/javascriptguide.xml) +- [Google JavaScript Style Guide](https://google.github.io/styleguide/jsguide.html) - [Airbnb JavaScript Style Guide](https://github.com/airbnb/javascript) - [Idiomatic.JS](https://github.com/rwaldron/idiomatic.js) - [StandardJS](https://standardjs.com/) - (plus many more) -If you're a novice developer, start with the cheatsheet at the beginning of this chapter. Once you've mastered that you can browse other style guides to pick up common principles and decide which one you like best. +If you're a novice developer, start with the cheat sheet at the beginning of this chapter. Then you can browse other style guides to pick up more ideas and decide which one you like best. ## Automated Linters -Linters are tools that can automatically check the style of your code and make suggestions for refactoring. +Linters are tools that can automatically check the style of your code and make improving suggestions. -The great thing about them is that style-checking can also find some bugs, like typos in variable or function names. Because of this feature, installing a linter is recommended even if you don't want to stick to one particular "code style". +The great thing about them is that style-checking can also find some bugs, like typos in variable or function names. Because of this feature, using a linter is recommended even if you don't want to stick to one particular "code style". -Here are the most well-known linting tools: +Here are some well-known linting tools: -- [JSLint](http://www.jslint.com/) -- one of the first linters. -- [JSHint](http://www.jshint.com/) -- more settings than JSLint. -- [ESLint](http://eslint.org/) -- probably the newest one. +- [JSLint](https://www.jslint.com/) -- one of the first linters. +- [JSHint](https://jshint.com/) -- more settings than JSLint. +- [ESLint](https://eslint.org/) -- probably the newest one. -All of them can do the job. The author uses [ESLint](http://eslint.org/). +All of them can do the job. The author uses [ESLint](https://eslint.org/). Most linters are integrated with many popular editors: just enable the plugin in the editor and configure the style. @@ -304,21 +328,21 @@ Here's an example of an `.eslintrc` file: }, "rules": { "no-console": 0, - }, - "indent": 2 + "indent": 2 + } } ``` Here the directive `"extends"` denotes that the configuration is based on the "eslint:recommended" set of settings. After that, we specify our own. -It is also possible to download style rule sets from the web and extend them instead. See <http://eslint.org/docs/user-guide/getting-started> for more details about installation. +It is also possible to download style rule sets from the web and extend them instead. See <https://eslint.org/docs/user-guide/getting-started> for more details about installation. Also certain IDEs have built-in linting, which is convenient but not as customizable as ESLint. ## Summary -All syntax rules described in this chapter (and in the style guides referenced) aim to increase the readability of your code, but all of them are debatable. +All syntax rules described in this chapter (and in the style guides referenced) aim to increase the readability of your code. All of them are debatable. -When we think about writing "better" code, the questions we should ask are, "What makes the code more readable and easier to understand?" and "What can help us avoid errors?" These are the main things to keep in mind when choosing and debating code styles. +When we think about writing "better" code, the questions we should ask ourselves are: "What makes the code more readable and easier to understand?" and "What can help us avoid errors?" These are the main things to keep in mind when choosing and debating code styles. Reading popular style guides will allow you to keep up to date with the latest ideas about code style trends and best practices. diff --git a/1-js/03-code-quality/02-coding-style/code-style.svg b/1-js/03-code-quality/02-coding-style/code-style.svg index 12a755c97..0a1db56c8 100644 --- a/1-js/03-code-quality/02-coding-style/code-style.svg +++ b/1-js/03-code-quality/02-coding-style/code-style.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="715" height="528" viewBox="0 0 715 528"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="code-style.svg"><path fill="#FFF" d="M0 0h715v528H0z"/><image id="Bitmap" width="395" height="340" x="239" y="143" xlink:href=""/><path id="Line-17" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 355.5v16"/><path id="Line-18" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M247.5 360.5l3-5"/><path id="Line-10" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l3-5"/><path id="Line-11" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M253.5 360.5l-3-5"/><path id="Line-12" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l-3-5"/><text id="2" fill="#C06334" font-family="OpenSans-Regular, Open Sans" font-size="12" font-weight="normal"><tspan x="249.069" y="186">2</tspan></text><path id="Line-19" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M259 190v-9.5"/><path id="Line-5" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190v-9.5"/><path id="Line-20" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190.5h14"/><text id="No-space-between-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="138.785" y="46">No space</tspan> <tspan x="13.007" y="65">between the function name and parentheses</tspan> <tspan x="13.263" y="84">between the parentheses and the parameter</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M279.5 93.086l.707.707 46.1 46.1 4.243-4.242 4.95 14.849-14.85-4.95 4.243-4.243-46.1-46.1-.707-.707 1.414-1.414z"/><text id="Indentation-2-spaces" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="91.04" y="168">Indentation</tspan> <tspan x="103.334" y="187">2 spaces</tspan></text><path id="Line-4" fill="#C06334" fill-rule="nonzero" d="M224 170.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="A-space-" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="105.48" y="214">A space </tspan> <tspan x="67.052" y="233">after for/if/while…</tspan></text><path id="Line-6" fill="#C06334" fill-rule="nonzero" d="M224 203.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="}-else-{-without-a-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="507.896" y="455">} else { without a line break</tspan></text><path id="Line-14" fill="#C06334" fill-rule="nonzero" d="M430.792 430.381l-.781 5.949 66.12 8.679.99.13-.26 1.983-.991-.13-66.12-8.679-.78 5.95L416 435.5l14.792-5.119z"/><text id="Spaces-around-a-nest" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="255.98" y="501">Spaces around a nested call</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M309 457.565l14.903 4.786-4.196 4.289 15.992 15.645.715.7-1.398 1.43-.715-.7-15.993-15.645-4.195 4.29L309 457.564z"/><text id="An-empty-line" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="68.465" y="362">An empty line</tspan> <tspan x="38.052" y="381">between logical blocks</tspan></text><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M218 354.5l14 7-14 7v-6h-39.558v-2H218v-6z"/><text id="Lines-are-not-very-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="515.101" y="350">Lines are not very long</tspan></text><path id="Line-13" fill="#C06334" fill-rule="nonzero" d="M544.024 354.211l.706 1.872-.935.353-78.344 29.557 2.119 5.614L452 390l10.628-11.491 2.117 5.613 78.344-29.558.935-.353z"/><text id="A-semicolon-;" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="609" y="223">A semicolon ;</tspan> <tspan x="609" y="242">is mandatory</tspan></text><path id="Line-8" fill="#C06334" fill-rule="nonzero" d="M409 221.5v6h192v2H409v6l-14-7 14-7z"/><text id="Spaces-around-operat" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="597.772" y="136">Spaces</tspan> <tspan x="559.973" y="155">around operators</tspan></text><path id="Line-16" fill="#C06334" fill-rule="nonzero" d="M585.27 163.314l.416 1.956-.978.208-160.307 34.088 1.249 5.869-15.15-3.935 12.238-9.759 1.247 5.868 160.307-34.087.978-.208z"/><text id="Curly-brace-{-on-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="526.676" y="70">Curly brace {</tspan> <tspan x="464.233" y="89">on the same line, after a space</tspan></text><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M527.484 97.671l.845 1.813-.907.422-105.311 49.086 2.536 5.438L409 154l9.732-12.26 2.535 5.439 105.31-49.085.907-.423z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M364.5 36v96h6l-7 14-7-14h6V36h2z"/><text id="A-space-between-argu" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="454.48" y="273">A space</tspan> <tspan x="450.532" y="292">between</tspan> <tspan x="442.91" y="311">arguments</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M434.94 285.587l1.473 1.353-.677.737-24.528 26.69 4.419 4.06L401 324l4.318-15.045 4.418 4.059 24.528-26.69.676-.737z"/><text id="A-space-between-para" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="270.969" y="20">A space between parameters</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M384.5 458.5l-6.869 14.065-3.645-4.767-22.879 17.496-.794.608-1.215-1.589.795-.607 22.878-17.497-3.644-4.765L384.5 458.5z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="715" height="528" viewBox="0 0 715 528"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="code-style.svg"><path fill="#FFF" d="M0 0h715v528H0z"/><image id="Bitmap" width="395" height="340" x="239" y="143" xlink:href=""/><path id="Line-17" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 355.5v16"/><path id="Line-18" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M247.5 360.5l3-5"/><path id="Line-10" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l3-5"/><path id="Line-11" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M253.5 360.5l-3-5"/><path id="Line-12" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l-3-5"/><text id="2" fill="#C06334" font-family="OpenSans-Regular, Open Sans" font-size="12" font-weight="normal"><tspan x="249.069" y="186">2</tspan></text><path id="Line-19" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M259 190v-9.5"/><path id="Line-5" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190v-9.5"/><path id="Line-20" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190.5h14"/><text id="No-space-between-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="138.785" y="46">No space</tspan> <tspan x="13.007" y="65">between the function name and parentheses</tspan> <tspan x="13.263" y="84">between the parentheses and the parameter</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M279.5 93.086l.707.707 46.1 46.1 4.243-4.242 4.95 14.849-14.85-4.95 4.243-4.243-46.1-46.1-.707-.707 1.414-1.414z"/><text id="Indentation-2-spaces" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="91.04" y="168">Indentation</tspan> <tspan x="103.334" y="187">2 spaces</tspan></text><path id="Line-4" fill="#C06334" fill-rule="nonzero" d="M224 170.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="A-space-" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="105.48" y="214">A space </tspan> <tspan x="67.052" y="233">after for/if/while…</tspan></text><path id="Line-6" fill="#C06334" fill-rule="nonzero" d="M224 203.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="}-else-{-without-a-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="507.896" y="455">} else { without a line break</tspan></text><path id="Line-14" fill="#C06334" fill-rule="nonzero" d="M430.792 430.381l-.781 5.949 66.12 8.679.99.13-.26 1.983-.991-.13-66.12-8.679-.78 5.95L416 435.5l14.792-5.119z"/><text id="Spaces-around-a-nest" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="255.98" y="501">Spaces around a nested call</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M309 457.565l14.903 4.786-4.196 4.289 15.992 15.645.715.7-1.398 1.43-.715-.7-15.993-15.645-4.195 4.29L309 457.564z"/><text id="An-empty-line" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="68.465" y="362">An empty line</tspan> <tspan x="38.052" y="381">between logical blocks</tspan></text><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M218 354.5l14 7-14 7v-6h-39.558v-2H218v-6z"/><text id="Lines-are-not-very-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="515.101" y="350">Lines are not very long</tspan></text><path id="Line-13" fill="#C06334" fill-rule="nonzero" d="M544.024 354.211l.706 1.872-.935.353-78.344 29.557 2.119 5.614L452 390l10.628-11.491 2.117 5.613 78.344-29.558.935-.353z"/><text id="A-semicolon-;" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="609" y="223">A semicolon ;</tspan> <tspan x="609" y="242">is mandatory</tspan></text><path id="Line-8" fill="#C06334" fill-rule="nonzero" d="M409 221.5v6h192v2H409v6l-14-7 14-7z"/><text id="Spaces-around-operat" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="597.772" y="136">Spaces</tspan> <tspan x="559.973" y="155">around operators</tspan></text><path id="Line-16" fill="#C06334" fill-rule="nonzero" d="M585.27 163.314l.416 1.956-.978.208-160.307 34.088 1.249 5.869-15.15-3.935 12.238-9.759 1.247 5.868 160.307-34.087.978-.208z"/><text id="Curly-brace-{-on-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="526.676" y="70">Curly brace {</tspan> <tspan x="464.233" y="89">on the same line, after a space</tspan></text><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M527.484 97.671l.845 1.813-.907.422-105.311 49.086 2.536 5.438L409 154l9.732-12.26 2.535 5.439 105.31-49.085.907-.423z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M364.5 36v96h6l-7 14-7-14h6V36h2z"/><text id="A-space-between-argu" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="454.48" y="273">A space</tspan> <tspan x="450.532" y="292">between</tspan> <tspan x="442.91" y="311">arguments</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M434.94 285.587l1.473 1.353-.677.737-24.528 26.69 4.419 4.06L401 324l4.318-15.045 4.418 4.059 24.528-26.69.676-.737z"/><text id="A-space-between-para" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="270.969" y="20">A space between parameters</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M384.5 458.5l-6.869 14.065-3.645-4.767-22.879 17.496-.794.608-1.215-1.589.795-.607 22.878-17.497-3.644-4.765L384.5 458.5z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="715" height="528" viewBox="0 0 715 528"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="code-style.svg"><path fill="#FFF" d="M0 0h715v528H0z"/><image id="Bitmap" width="395" height="340" x="239" y="143" xlink:href=""/><path id="Line-17" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 355.5v16"/><path id="Line-18" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M247.5 360.5l3-5"/><path id="Line-10" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l3-5"/><path id="Line-11" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M253.5 360.5l-3-5"/><path id="Line-12" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M250.5 372.5l-3-5"/><text id="2" fill="#C06334" font-family="OpenSans-Regular, Open Sans" font-size="12" font-weight="normal"><tspan x="249.069" y="186">2</tspan></text><path id="Line-19" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M259 190v-9.5"/><path id="Line-5" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190v-9.5"/><path id="Line-20" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M245 190.5h14"/><text id="No-space-between-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="138.785" y="46">No space</tspan> <tspan x="13.007" y="65">between the function name and parentheses</tspan> <tspan x="13.263" y="84">between the parentheses and the parameter</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M279.5 93.086l.707.707 46.1 46.1 4.243-4.242 4.95 14.849-14.85-4.95 4.243-4.243-46.1-46.1-.707-.707 1.414-1.414z"/><text id="Indentation-2-spaces" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="91.04" y="168">Indentation</tspan> <tspan x="103.334" y="187">2 spaces</tspan></text><path id="Line-4" fill="#C06334" fill-rule="nonzero" d="M224 170.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="A-space-" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="105.48" y="214">A space </tspan> <tspan x="67.052" y="233">after for/if/while…</tspan></text><path id="Line-6" fill="#C06334" fill-rule="nonzero" d="M224 203.5l14 7-14 7v-6h-46.5v-2H224v-6z"/><text id="}-else-{-without-a-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="507.896" y="455">} else { without a line break</tspan></text><path id="Line-14" fill="#C06334" fill-rule="nonzero" d="M430.792 430.381l-.781 5.949 66.12 8.679.99.13-.26 1.983-.991-.13-66.12-8.679-.78 5.95L416 435.5l14.792-5.119z"/><text id="Spaces-around-a-nest" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="255.98" y="501">Spaces around a nested call</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M309 457.565l14.903 4.786-4.196 4.289 15.992 15.645.715.7-1.398 1.43-.715-.7-15.993-15.645-4.195 4.29L309 457.564z"/><text id="An-empty-line" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="68.465" y="362">An empty line</tspan> <tspan x="38.052" y="381">between logical blocks</tspan></text><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M218 354.5l14 7-14 7v-6h-39.558v-2H218v-6z"/><text id="Lines-are-not-very-l" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="515.101" y="350">Lines are not very long</tspan></text><path id="Line-13" fill="#C06334" fill-rule="nonzero" d="M544.024 354.211l.706 1.872-.935.353-78.344 29.557 2.119 5.614L452 390l10.628-11.491 2.117 5.613 78.344-29.558.935-.353z"/><text id="A-semicolon-;" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="609" y="223">A semicolon ;</tspan> <tspan x="609" y="242">is mandatory</tspan></text><path id="Line-8" fill="#C06334" fill-rule="nonzero" d="M409 221.5v6h192v2H409v6l-14-7 14-7z"/><text id="Spaces-around-operat" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="597.772" y="136">Spaces</tspan> <tspan x="559.973" y="155">around operators</tspan></text><path id="Line-16" fill="#C06334" fill-rule="nonzero" d="M585.27 163.314l.416 1.956-.978.208-160.307 34.088 1.249 5.869-15.15-3.935 12.238-9.759 1.247 5.868 160.307-34.087.978-.208z"/><text id="Curly-brace-{-on-the" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="526.676" y="70">Curly brace {</tspan> <tspan x="464.233" y="89">on the same line, after a space</tspan></text><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M527.484 97.671l.845 1.813-.907.422-105.311 49.086 2.536 5.438L409 154l9.732-12.26 2.535 5.439 105.31-49.085.907-.423z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M364.5 36v96h6l-7 14-7-14h6V36h2z"/><text id="A-space-between-argu" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="454.48" y="273">A space</tspan> <tspan x="450.532" y="292">between</tspan> <tspan x="442.91" y="311">arguments</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M434.94 285.587l1.473 1.353-.677.737-24.528 26.69 4.419 4.06L401 324l4.318-15.045 4.418 4.059 24.528-26.69.676-.737z"/><text id="A-space-between-para" fill="#C06334" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="270.969" y="20">A space between parameters</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M384.5 458.5l-6.869 14.065-3.645-4.767-22.879 17.496-.794.608-1.215-1.589.795-.607 22.878-17.497-3.644-4.765L384.5 458.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/03-code-quality/02-coding-style/figure-bracket-style.png b/1-js/03-code-quality/02-coding-style/figure-bracket-style.png deleted file mode 100644 index b04db65c6..000000000 Binary files a/1-js/03-code-quality/02-coding-style/figure-bracket-style.png and /dev/null differ diff --git a/1-js/03-code-quality/02-coding-style/figure-bracket-style@2x.png b/1-js/03-code-quality/02-coding-style/figure-bracket-style@2x.png deleted file mode 100644 index 0e994ca4b..000000000 Binary files a/1-js/03-code-quality/02-coding-style/figure-bracket-style@2x.png and /dev/null differ diff --git a/1-js/03-code-quality/03-comments/article.md b/1-js/03-code-quality/03-comments/article.md index 930ff929f..af3a06c80 100644 --- a/1-js/03-code-quality/03-comments/article.md +++ b/1-js/03-code-quality/03-comments/article.md @@ -4,7 +4,7 @@ As we know from the chapter <info:structure>, comments can be single-line: start We normally use them to describe how and why the code works. -From the first sight, commenting might be obvious, but novices in programming usually get it wrong. +At first sight, commenting might be obvious, but novices in programming often use them wrongly. ## Bad comments @@ -18,7 +18,7 @@ complex; code; ``` -But in good code the amount of such "explanatory" comments should be minimal. Seriously, code should be easy to understand without them. +But in good code, the amount of such "explanatory" comments should be minimal. Seriously, the code should be easy to understand without them. There's a great rule about that: "if the code is so unclear that it requires a comment, then maybe it should be rewritten instead". @@ -120,30 +120,30 @@ In reality, we can't totally avoid "explanatory" comments. There are complex alg So, explanatory comments are usually bad. Which comments are good? Describe the architecture -: Provide a high-level overview of components, how they interact, what's the control flow in various situations... In short -- the bird's eye view of the code. There's a special diagram language [UML](http://wikipedia.org/wiki/Unified_Modeling_Language) for high-level architecture diagrams. Definitely worth studying. +: Provide a high-level overview of components, how they interact, what's the control flow in various situations... In short -- the bird's eye view of the code. There's a special language [UML](http://wikipedia.org/wiki/Unified_Modeling_Language) to build high-level architecture diagrams explaining the code. Definitely worth studying. -Document a function usage +Document function parameters and usage : There's a special syntax [JSDoc](http://en.wikipedia.org/wiki/JSDoc) to document a function: usage, parameters, returned value. - For instance: - ```js - /** - * Returns x raised to the n-th power. - * - * @param {number} x The number to raise. - * @param {number} n The power, must be a natural number. - * @return {number} x raised to the n-th power. - */ - function pow(x, n) { - ... - } - ``` +For instance: +```js +/** + * Returns x raised to the n-th power. + * + * @param {number} x The number to raise. + * @param {number} n The power, must be a natural number. + * @return {number} x raised to the n-th power. + */ +function pow(x, n) { + ... +} +``` - Such comments allow us to understand the purpose of the function and use it the right way without looking in its code. +Such comments allow us to understand the purpose of the function and use it the right way without looking in its code. - By the way, many editors like [WebStorm](https://www.jetbrains.com/webstorm/) can understand them as well and use them to provide autocomplete and some automatic code-checking. +By the way, many editors like [WebStorm](https://www.jetbrains.com/webstorm/) can understand them as well and use them to provide autocomplete and some automatic code-checking. - Also, there are tools like [JSDoc 3](https://github.com/jsdoc3/jsdoc) that can generate HTML-documentation from the comments. You can read more information about JSDoc at <http://usejsdoc.org/>. +Also, there are tools like [JSDoc 3](https://github.com/jsdoc/jsdoc) that can generate HTML-documentation from the comments. You can read more information about JSDoc at <https://jsdoc.app>. Why is the task solved this way? : What's written is important. But what's *not* written may be even more important to understand what's going on. Why is the task solved exactly this way? The code gives no answer. @@ -175,6 +175,6 @@ Good comments allow us to maintain the code well, come back to it after a delay **Avoid comments:** - That tell "how code works" and "what it does". -- Put them only if it's impossible to make the code so simple and self-descriptive that it doesn't require those. +- Put them in only if it's impossible to make the code so simple and self-descriptive that it doesn't require them. Comments are also used for auto-documenting tools like JSDoc3: they read them and generate HTML-docs (or docs in another format). diff --git a/1-js/03-code-quality/04-ninja-code/article.md b/1-js/03-code-quality/04-ninja-code/article.md index 9019242f2..96fdf4143 100644 --- a/1-js/03-code-quality/04-ninja-code/article.md +++ b/1-js/03-code-quality/04-ninja-code/article.md @@ -1,7 +1,7 @@ # Ninja code -```quote author="Confucius" +```quote author="Confucius (Analects)" Learning without thought is labor lost; thought without learning is perilous. ``` @@ -43,7 +43,7 @@ The Dao hides in wordlessness. Only the Dao is well begun and well completed. ``` -Another way to code faster is to use single-letter variable names everywhere. Like `a`, `b` or `c`. +Another way to code shorter is to use single-letter variable names everywhere. Like `a`, `b` or `c`. A short variable disappears in the code like a real ninja in the forest. No one will be able to find it using "search" of the editor. And even if someone does, they won't be able to "decipher" what the name `a` or `b` means. @@ -104,8 +104,8 @@ A quick read of such code becomes impossible. And when there's a typo... Ummm... ## Smart synonyms -```quote author="Confucius" -The hardest thing of all is to find a black cat in a dark room, especially if there is no cat. +```quote author="Laozi (Tao Te Ching)" +The Tao that can be told is not the eternal Tao. The name that can be named is not the eternal name. ``` Using *similar* names for *same* things makes life more interesting and shows your creativity to the public. @@ -137,7 +137,7 @@ Instead, reuse existing names. Just write new values into them. In a function try to use only variables passed as parameters. -That would make it really hard to identify what's exactly in the variable *now*. And also where it comes from. A person with weak intuition would have to analyze the code line-by-line and track the changes through every code branch. +That would make it really hard to identify what's exactly in the variable *now*. And also where it comes from. The purpose is to develop the intuition and memory of a person reading the code. A person with weak intuition would have to analyze the code line-by-line and track the changes through every code branch. **An advanced variant of the approach is to covertly (!) replace the value with something alike in the middle of a loop or a function.** @@ -155,7 +155,7 @@ function ninjaFunction(elem) { A fellow programmer who wants to work with `elem` in the second half of the function will be surprised... Only during the debugging, after examining the code they will find out that they're working with a clone! -Seen in code regularly. Deadly effective even against an experienced ninja. +Seen in code regularly. Deadly effective even against an experienced ninja. ## Underscores for fun @@ -169,8 +169,7 @@ A smart ninja puts underscores at one spot of code and evades them at other plac Let everyone see how magnificent your entities are! Names like `superElement`, `megaFrame` and `niceItem` will definitely enlighten a reader. -Indeed, from one hand, something is written: `super..`, `mega..`, `nice..` But from the other hand -- that brings no details. A reader may decide to look for a hidden meaning and meditate for an hour or two. - +Indeed, from one hand, something is written: `super..`, `mega..`, `nice..` But from the other hand -- that brings no details. A reader may decide to look for a hidden meaning and meditate for an hour or two of their paid working time. ## Overlap outer variables @@ -180,7 +179,7 @@ When in the light, can't see anything in the darkness.<br> When in the darkness, can see everything in the light. ``` -Use same names for variables inside and outside a function. As simple. No efforts required. +Use same names for variables inside and outside a function. As simple. No efforts to invent new names. ```js let *!*user*/!* = authenticateUser(); diff --git a/1-js/03-code-quality/05-testing-mocha/3-pow-test-wrong/solution.md b/1-js/03-code-quality/05-testing-mocha/3-pow-test-wrong/solution.md index 7b58f0bf1..4d0571b9d 100644 --- a/1-js/03-code-quality/05-testing-mocha/3-pow-test-wrong/solution.md +++ b/1-js/03-code-quality/05-testing-mocha/3-pow-test-wrong/solution.md @@ -4,7 +4,7 @@ What we have here is actually 3 tests, but layed out as a single function with 3 Sometimes it's easier to write this way, but if an error occurs, it's much less obvious what went wrong. -If an error happens inside a complex execution flow, then we'll have to figure out the data at that point. We'll actually have to *debug the test*. +If an error happens in the middle of a complex execution flow, then we'll have to figure out the data at that point. We'll actually have to *debug the test*. It would be much better to break the test into multiple `it` blocks with clearly written inputs and outputs. diff --git a/1-js/03-code-quality/05-testing-mocha/article.md b/1-js/03-code-quality/05-testing-mocha/article.md index c7b164b15..4c2b1aa5e 100644 --- a/1-js/03-code-quality/05-testing-mocha/article.md +++ b/1-js/03-code-quality/05-testing-mocha/article.md @@ -1,10 +1,8 @@ -# Automated testing with mocha +# Automated testing with Mocha -Automated testing will be used in further tasks. +Automated testing will be used in further tasks, and it's also widely used in real projects. -It's actually a part of the "educational minimum" of a developer. - -## Why we need tests? +## Why do we need tests? When we write a function, we can usually imagine what it should do: which parameters give which results. @@ -20,15 +18,15 @@ For instance, we're creating a function `f`. Wrote some code, testing: `f(1)` wo That's very typical. When we develop something, we keep a lot of possible use cases in mind. But it's hard to expect a programmer to check all of them manually after every change. So it becomes easy to fix one thing and break another one. -**Automated testing means that tests are written separately, in addition to the code. They can be executed easily and check all the main use cases.** +**Automated testing means that tests are written separately, in addition to the code. They run our functions in various ways and compare results with the expected.** ## Behavior Driven Development (BDD) -Let's use a technique named [Behavior Driven Development](http://en.wikipedia.org/wiki/Behavior-driven_development) or, in short, BDD. That approach is used among many projects. BDD is not just about testing. That's more. +Let's start with a technique named [Behavior Driven Development](http://en.wikipedia.org/wiki/Behavior-driven_development) or, in short, BDD. **BDD is three things in one: tests AND documentation AND examples.** -Enough words. Let's see the example. +To understand BDD, we'll examine a practical case of development. ## Development of "pow": the spec @@ -38,7 +36,7 @@ That task is just an example: there's the `**` operator in JavaScript that can d Before creating the code of `pow`, we can imagine what the function should do and describe it. -Such description is called a *specification* or, in short, a spec, and looks like this: +Such description is called a *specification* or, in short, a spec, and contains descriptions of use cases together with tests for them, like this: ```js describe("pow", function() { @@ -53,17 +51,17 @@ describe("pow", function() { A spec has three main building blocks that you can see above: `describe("title", function() { ... })` -: What functionality we're describing. Uses to group "workers" -- the `it` blocks. In our case we're describing the function `pow`. +: What functionality we're describing? In our case we're describing the function `pow`. Used to group "workers" -- the `it` blocks. -`it("title", function() { ... })` +`it("use case description", function() { ... })` : In the title of `it` we *in a human-readable way* describe the particular use case, and the second argument is a function that tests it. `assert.equal(value1, value2)` : The code inside `it` block, if the implementation is correct, should execute without errors. - Functions `assert.*` are used to check whether `pow` works as expected. Right here we're using one of them -- `assert.equal`, it compares arguments and yields an error if they are not equal. Here it checks that the result of `pow(2, 3)` equals `8`. + Functions `assert.*` are used to check whether `pow` works as expected. Right here we're using one of them -- `assert.equal`, it compares arguments and yields an error if they are not equal. Here it checks that the result of `pow(2, 3)` equals `8`. There are other types of comparisons and checks, that we'll add later. - There are other types of comparisons and checks that we'll see further. +The specification can be executed, and it will run the test specified in `it` block. We'll see that later. ## The development flow @@ -71,7 +69,7 @@ The flow of development usually looks like this: 1. An initial spec is written, with tests for the most basic functionality. 2. An initial implementation is created. -3. To check whether it works, we run the testing framework [Mocha](http://mochajs.org/) (more details soon) that runs the spec. Errors are displayed. We make corrections until everything works. +3. To check whether it works, we run the testing framework [Mocha](https://mochajs.org/) (more details soon) that runs the spec. While the functionality is not complete, errors are displayed. We make corrections until everything works. 4. Now we have a working initial implementation with tests. 5. We add more use cases to the spec, probably not yet supported by the implementations. Tests start to fail. 6. Go to 3, update the implementation till tests give no errors. @@ -79,15 +77,17 @@ The flow of development usually looks like this: So, the development is *iterative*. We write the spec, implement it, make sure tests pass, then write more tests, make sure they work etc. At the end we have both a working implementation and tests for it. -In our case, the first step is complete: we have an initial spec for `pow`. So let's make an implementation. But before that let's make a "zero" run of the spec, just to see that tests are working (they will all fail). +Let's see this development flow in our practical case. + +The first step is already complete: we have an initial spec for `pow`. Now, before making the implementation, let's use a few JavaScript libraries to run the tests, just to see that they are working (they will all fail). ## The spec in action Here in the tutorial we'll be using the following JavaScript libraries for tests: -- [Mocha](http://mochajs.org/) -- the core framework: it provides common testing functions including `describe` and `it` and the main function that runs tests. -- [Chai](http://chaijs.com) -- the library with many assertions. It allows to use a lot of different assertions, for now we need only `assert.equal`. -- [Sinon](http://sinonjs.org/) -- a library to spy over functions, emulate built-in functions and more, we'll need it much later. +- [Mocha](https://mochajs.org/) -- the core framework: it provides common testing functions including `describe` and `it` and the main function that runs tests. +- [Chai](https://www.chaijs.com/) -- the library with many assertions. It allows to use a lot of different assertions, for now we need only `assert.equal`. +- [Sinon](https://sinonjs.org/) -- a library to spy over functions, emulate built-in functions and more, we'll need it much later. These libraries are suitable for both in-browser and server-side testing. Here we'll consider the browser variant. @@ -110,7 +110,7 @@ The result: As of now, the test fails, there's an error. That's logical: we have an empty function code in `pow`, so `pow(2,3)` returns `undefined` instead of `8`. -For the future, let's note that there are advanced test-runners, like [karma](https://karma-runner.github.io/) and others. So it's generally not a problem to setup many different tests. +For the future, let's note that there are more high-level test-runners, like [karma](https://karma-runner.github.io/) and others, that make it easy to autorun many different tests. ## Initial implementation @@ -132,7 +132,7 @@ What we've done is definitely a cheat. The function does not work: an attempt to ...But the situation is quite typical, it happens in practice. Tests pass, but the function works wrong. Our spec is imperfect. We need to add more use cases to it. -Let's add one more test to see if `pow(3, 4) = 81`. +Let's add one more test to check that `pow(3, 4) = 81`. We can select one of two ways to organize the test here: @@ -159,8 +159,8 @@ We can select one of two ways to organize the test here: assert.equal(pow(2, 3), 8); }); - it("3 raised to power 3 is 27", function() { - assert.equal(pow(3, 3), 27); + it("3 raised to power 4 is 81", function() { + assert.equal(pow(3, 4), 81); }); }); @@ -182,7 +182,7 @@ The result: [iframe height=250 src="pow-2" edit border="1"] -As we could expect, the second test failed. Sure, our function always returns `8`, while the `assert` expects `27`. +As we could expect, the second test failed. Sure, our function always returns `8`, while the `assert` expects `81`. ## Improving the implementation @@ -296,7 +296,7 @@ Testing finished – after all tests (after) [edit src="beforeafter" title="Open the example in the sandbox."] -Usually, `beforeEach/afterEach` (`before/after`) are used to perform initialization, zero out counters or do something else between the tests (or test groups). +Usually, `beforeEach/afterEach` and `before/after` are used to perform initialization, zero out counters or do something else between the tests (or test groups). ```` ## Extending the spec @@ -336,17 +336,16 @@ The result with new tests: The newly added tests fail, because our implementation does not support them. That's how BDD is done: first we write failing tests, and then make an implementation for them. ```smart header="Other assertions" - Please note the assertion `assert.isNaN`: it checks for `NaN`. -There are other assertions in Chai as well, for instance: +There are other assertions in [Chai](https://www.chaijs.com/) as well, for instance: - `assert.equal(value1, value2)` -- checks the equality `value1 == value2`. - `assert.strictEqual(value1, value2)` -- checks the strict equality `value1 === value2`. - `assert.notEqual`, `assert.notStrictEqual` -- inverse checks to the ones above. - `assert.isTrue(value)` -- checks that `value === true` - `assert.isFalse(value)` -- checks that `value === false` -- ...the full list is in the [docs](http://chaijs.com/api/assert/) +- ...the full list is in the [docs](https://www.chaijs.com/api/assert/) ``` So we should add a couple of lines to `pow`: @@ -380,9 +379,9 @@ In BDD, the spec goes first, followed by implementation. At the end we have both The spec can be used in three ways: -1. **Tests** guarantee that the code works correctly. -2. **Docs** -- the titles of `describe` and `it` tell what the function does. -3. **Examples** -- the tests are actually working examples showing how a function can be used. +1. As **Tests** - they guarantee that the code works correctly. +2. As **Docs** -- the titles of `describe` and `it` tell what the function does. +3. As **Examples** -- the tests are actually working examples showing how a function can be used. With the spec, we can safely improve, change, even rewrite the function from scratch and make sure it still works right. @@ -390,23 +389,21 @@ That's especially important in large projects when a function is used in many pl Without tests, people have two ways: -1. To perform the change, no matter what. And then our users meet bugs and report them. If we can afford that. -2. Or people become afraid to modify such functions, if the punishment for errors is harsh. Then it becomes old, overgrown with cobwebs, no one wants to get into it, and that's not good. +1. To perform the change, no matter what. And then our users meet bugs, as we probably fail to check something manually. +2. Or, if the punishment for errors is harsh, as there are no tests, people become afraid to modify such functions, and then the code becomes outdated, no one wants to get into it. Not good for development. -**Automatically tested code is contrary to that!** +**Automatic testing helps to avoid these problems!** -If the project is covered with tests, there's just no such problem. We can run tests and see a lot of checks made in a matter of seconds. +If the project is covered with tests, there's just no such problem. After any changes, we can run tests and see a lot of checks made in a matter of seconds. **Besides, a well-tested code has better architecture.** -Naturally, that's because it's easier to change and improve it. But not only that. +Naturally, that's because auto-tested code is easier to modify and improve. But there's also another reason. To write tests, the code should be organized in such a way that every function has a clearly described task, well-defined input and output. That means a good architecture from the beginning. In real life that's sometimes not that easy. Sometimes it's difficult to write a spec before the actual code, because it's not yet clear how it should behave. But in general writing tests makes development faster and more stable. -## What now? - Later in the tutorial you will meet many tasks with tests baked-in. So you'll see more practical examples. Writing tests requires good JavaScript knowledge. But we're just starting to learn it. So, to settle down everything, as of now you're not required to write tests, but you should already be able to read them even if they are a little bit more complex than in this chapter. diff --git a/1-js/03-code-quality/05-testing-mocha/beforeafter.view/test.js b/1-js/03-code-quality/05-testing-mocha/beforeafter.view/test.js index cad51d3ee..d3de82546 100644 --- a/1-js/03-code-quality/05-testing-mocha/beforeafter.view/test.js +++ b/1-js/03-code-quality/05-testing-mocha/beforeafter.view/test.js @@ -1,5 +1,11 @@ describe("test", function() { + + // Mocha usually waits for the tests for 2 seconds before considering them wrong + + this.timeout(200000); // With this code we increase this - in this case to 200,000 milliseconds + // This is because of the "alert" function, because if you delay pressing the "OK" button the tests will not pass! + before(() => alert("Testing started – before all tests")); after(() => alert("Testing finished – after all tests")); diff --git a/1-js/03-code-quality/05-testing-mocha/pow-2.view/test.js b/1-js/03-code-quality/05-testing-mocha/pow-2.view/test.js index 9a2f8fde7..c803f0e61 100644 --- a/1-js/03-code-quality/05-testing-mocha/pow-2.view/test.js +++ b/1-js/03-code-quality/05-testing-mocha/pow-2.view/test.js @@ -4,8 +4,8 @@ describe("pow", function() { assert.equal(pow(2, 3), 8); }); - it("3 raised to power 3 is 27", function() { - assert.equal(pow(3, 3), 27); + it("3 raised to power 4 is 81", function() { + assert.equal(pow(3, 4), 81); }); }); diff --git a/1-js/03-code-quality/06-polyfills/article.md b/1-js/03-code-quality/06-polyfills/article.md index 907730fdc..5ca123908 100644 --- a/1-js/03-code-quality/06-polyfills/article.md +++ b/1-js/03-code-quality/06-polyfills/article.md @@ -1,57 +1,89 @@ -# Polyfills +# Polyfills and transpilers -The JavaScript language steadily evolves. New proposals to the language appear regularly, they are analyzed and, if considered worthy, are appended to the list at <https://tc39.github.io/ecma262/> and then progress to the [specification](http://www.ecma-international.org/publications/standards/Ecma-262.htm). +The JavaScript language steadily evolves. New proposals to the language appear regularly, they are analyzed and, if considered worthy, are appended to the list at <https://tc39.github.io/ecma262/> and then progress to the [specification](https://www.ecma-international.org/publications-and-standards/standards/ecma-262/). Teams behind JavaScript engines have their own ideas about what to implement first. They may decide to implement proposals that are in draft and postpone things that are already in the spec, because they are less interesting or just harder to do. -So it's quite common for an engine to implement only the part of the standard. +So it's quite common for an engine to implement only part of the standard. -A good page to see the current state of support for language features is <https://kangax.github.io/compat-table/es6/> (it's big, we have a lot to study yet). +A good page to see the current state of support for language features is <https://compat-table.github.io/compat-table/es6/> (it's big, we have a lot to study yet). -## Babel +As programmers, we'd like to use most recent features. The more good stuff - the better! -When we use modern features of the language, some engines may fail to support such code. Just as said, not all features are implemented everywhere. +On the other hand, how to make our modern code work on older engines that don't understand recent features yet? -Here Babel comes to the rescue. +There are two tools for that: -[Babel](https://babeljs.io) is a [transpiler](https://en.wikipedia.org/wiki/Source-to-source_compiler). It rewrites modern JavaScript code into the previous standard. +1. Transpilers. +2. Polyfills. -Actually, there are two parts in Babel: +Here, in this chapter, our purpose is to get the gist of how they work, and their place in web development. -1. First, the transpiler program, which rewrites the code. The developer runs it on their own computer. It rewrites the code into the older standard. And then the code is delivered to the website for users. Modern project build system like [webpack](http://webpack.github.io/) or [brunch](http://brunch.io/) provide means to run transpiler automatically on every code change, so that doesn't involve any time loss from our side. +## Transpilers -2. Second, the polyfill. +A [transpiler](https://en.wikipedia.org/wiki/Source-to-source_compiler) is a special piece of software that translates source code to another source code. It can parse ("read and understand") modern code and rewrite it using older syntax constructs, so that it'll also work in outdated engines. - The transpiler rewrites the code, so syntax features are covered. But for new functions we need to write a special script that implements them. JavaScript is a highly dynamic language, scripts may not just add new functions, but also modify built-in ones, so that they behave according to the modern standard. +E.g. JavaScript before year 2020 didn't have the "nullish coalescing operator" `??`. So, if a visitor uses an outdated browser, it may fail to understand the code like `height = height ?? 100`. - There's a term "polyfill" for scripts that "fill in" the gap and add missing implementations. +A transpiler would analyze our code and rewrite `height ?? 100` into `(height !== undefined && height !== null) ? height : 100`. - Two interesting polyfills are: - - [babel polyfill](https://babeljs.io/docs/usage/polyfill/) that supports a lot, but is big. - - [polyfill.io](http://polyfill.io) service that allows to load/construct polyfills on-demand, depending on the features we need. +```js +// before running the transpiler +height = height ?? 100; -So, we need to setup the transpiler and add the polyfill for old engines to support modern features. +// after running the transpiler +height = (height !== undefined && height !== null) ? height : 100; +``` -If we orient towards modern engines and do not use features except those supported everywhere, then we don't need to use Babel. +Now the rewritten code is suitable for older JavaScript engines. -## Examples in the tutorial +Usually, a developer runs the transpiler on their own computer, and then deploys the transpiled code to the server. +Speaking of names, [Babel](https://babeljs.io) is one of the most prominent transpilers out there. -````online -Most examples are runnable at-place, like this: +Modern project build systems, such as [webpack](https://webpack.js.org/), provide a means to run a transpiler automatically on every code change, so it's very easy to integrate into the development process. -```js run -alert('Press the "Play" button in the upper-right corner to run'); -``` +## Polyfills + +New language features may include not only syntax constructs and operators, but also built-in functions. + +For example, `Math.trunc(n)` is a function that "cuts off" the decimal part of a number, e.g `Math.trunc(1.23)` returns `1`. + +In some (very outdated) JavaScript engines, there's no `Math.trunc`, so such code will fail. + +As we're talking about new functions, not syntax changes, there's no need to transpile anything here. We just need to declare the missing function. + +A script that updates/adds new functions is called "polyfill". It "fills in" the gap and adds missing implementations. -Examples that use modern JS will work only if your browser supports it. -```` +For this particular case, the polyfill for `Math.trunc` is a script that implements it, like this: -```offline -As you're reading the offline version, examples are not runnable. But they usually work :) +```js +if (!Math.trunc) { // if no such function + // implement it + Math.trunc = function(number) { + // Math.ceil and Math.floor exist even in ancient JavaScript engines + // they are covered later in the tutorial + return number < 0 ? Math.ceil(number) : Math.floor(number); + }; +} ``` -[Chrome Canary](https://www.google.com/chrome/browser/canary.html) is good for all examples, but other modern browsers are mostly fine too. +JavaScript is a highly dynamic language. Scripts may add/modify any function, even built-in ones. + +One interesting polyfill library is [core-js](https://github.com/zloirock/core-js), which supports a wide range of features and allows you to include only the ones you need. + +## Summary + +In this chapter we'd like to motivate you to study modern and even "bleeding-edge" language features, even if they aren't yet well-supported by JavaScript engines. + +Just don't forget to use a transpiler (if using modern syntax or operators) and polyfills (to add functions that may be missing). They'll ensure that the code works. + +For example, later when you're familiar with JavaScript, you can setup a code build system based on [webpack](https://webpack.js.org/) with the [babel-loader](https://github.com/babel/babel-loader) plugin. + +Good resources that show the current state of support for various features: +- <https://compat-table.github.io/compat-table/es6/> - for pure JavaScript. +- <https://caniuse.com/> - for browser-related functions. + +P.S. Google Chrome is usually the most up-to-date with language features, try it if a tutorial demo fails. Most tutorial demos work with any modern browser though. -Note that on production we can use Babel to translate the code into suitable for less recent browsers, so there will be no such limitation, the code will run everywhere. diff --git a/1-js/04-object-basics/01-object/4-const-object/solution.md b/1-js/04-object-basics/01-object/4-const-object/solution.md deleted file mode 100644 index f73c2f92b..000000000 --- a/1-js/04-object-basics/01-object/4-const-object/solution.md +++ /dev/null @@ -1,19 +0,0 @@ -Sure, it works, no problem. - -The `const` only protects the variable itself from changing. - -In other words, `user` stores a reference to the object. And it can't be changed. But the content of the object can. - -```js run -const user = { - name: "John" -}; - -*!* -// works -user.name = "Pete"; -*/!* - -// error -user = 123; -``` diff --git a/1-js/04-object-basics/01-object/4-const-object/task.md b/1-js/04-object-basics/01-object/4-const-object/task.md deleted file mode 100644 index a9aada631..000000000 --- a/1-js/04-object-basics/01-object/4-const-object/task.md +++ /dev/null @@ -1,18 +0,0 @@ -importance: 5 - ---- - -# Constant objects? - -Is it possible to change an object declared with `const`? What do you think? - -```js -const user = { - name: "John" -}; - -*!* -// does it work? -user.name = "Pete"; -*/!* -``` diff --git a/1-js/04-object-basics/01-object/8-multiply-numeric/task.md b/1-js/04-object-basics/01-object/8-multiply-numeric/task.md index 33eb89220..6878ca088 100644 --- a/1-js/04-object-basics/01-object/8-multiply-numeric/task.md +++ b/1-js/04-object-basics/01-object/8-multiply-numeric/task.md @@ -2,9 +2,9 @@ importance: 3 --- -# Multiply numeric properties by 2 +# Multiply numeric property values by 2 -Create a function `multiplyNumeric(obj)` that multiplies all numeric properties of `obj` by `2`. +Create a function `multiplyNumeric(obj)` that multiplies all numeric property values of `obj` by `2`. For instance: diff --git a/1-js/04-object-basics/01-object/article.md b/1-js/04-object-basics/01-object/article.md index f59ec0292..f044e0563 100644 --- a/1-js/04-object-basics/01-object/article.md +++ b/1-js/04-object-basics/01-object/article.md @@ -1,7 +1,7 @@ # Objects -As we know from the chapter <info:types>, there are seven data types in JavaScript. Six of them are called "primitive", because their values contain only a single thing (be it a string or a number or whatever). +As we know from the chapter <info:types>, there are eight data types in JavaScript. Seven of them are called "primitive", because their values contain only a single thing (be it a string or a number or whatever). In contrast, objects are used to store keyed collections of various data and more complex entities. In JavaScript, objects penetrate almost every aspect of the language. So we must understand them first before going in-depth anywhere else. @@ -44,12 +44,12 @@ The resulting `user` object can be imagined as a cabinet with two signed files l  -We can add, remove and read files from it any time. +We can add, remove and read files from it at any time. Property values are accessible using the dot notation: ```js -// get fields of the object: +// get property values of the object: alert( user.name ); // John alert( user.age ); // 30 ``` @@ -62,7 +62,7 @@ user.isAdmin = true;  -To remove a property, we can use `delete` operator: +To remove a property, we can use the `delete` operator: ```js delete user.age; @@ -101,10 +101,11 @@ For multiword properties, the dot access doesn't work: user.likes birds = true ``` -That's because the dot requires the key to be a valid variable identifier. That is: no spaces and other limitations. +JavaScript doesn't understand that. It thinks that we address `user.likes`, and then gives a syntax error when comes across unexpected `birds`. -There's an alternative "square bracket notation" that works with any string: +The dot requires the key to be a valid variable identifier. That implies: contains no spaces, doesn't start with a digit and doesn't include special characters (`$` and `_` are allowed). +There's an alternative "square bracket notation" that works with any string: ```js run let user = {}; @@ -130,7 +131,7 @@ let key = "likes birds"; user[key] = true; ``` -Here, the variable `key` may be calculated at run-time or depend on the user input. And then we use it to access the property. That gives us a great deal of flexibility. The dot notation cannot be used in a similar way. +Here, the variable `key` may be calculated at run-time or depend on the user input. And then we use it to access the property. That gives us a great deal of flexibility. For instance: @@ -146,10 +147,21 @@ let key = prompt("What do you want to know about the user?", "name"); alert( user[key] ); // John (if enter "name") ``` +The dot notation cannot be used in a similar way: + +```js run +let user = { + name: "John", + age: 30 +}; + +let key = "name"; +alert( user.key ) // undefined +``` ### Computed properties -We can use square brackets in an object literal. That's called *computed properties*. +We can use square brackets in an object literal, when creating an object. That's called *computed properties*. For instance: @@ -189,49 +201,13 @@ let bag = { }; ``` -Square brackets are much more powerful than the dot notation. They allow any property names and variables. But they are also more cumbersome to write. +Square brackets are much more powerful than dot notation. They allow any property names and variables. But they are also more cumbersome to write. So most of the time, when property names are known and simple, the dot is used. And if we need something more complex, then we switch to square brackets. - - -````smart header="Reserved words are allowed as property names" -A variable cannot have a name equal to one of language-reserved words like "for", "let", "return" etc. - -But for an object property, there's no such restriction. Any name is fine: - -```js run -let obj = { - for: 1, - let: 2, - return: 3 -}; - -alert( obj.for + obj.let + obj.return ); // 6 -``` - -Basically, any name is allowed, but there's a special one: `"__proto__"` that gets special treatment for historical reasons. For instance, we can't set it to a non-object value: - -```js run -let obj = {}; -obj.__proto__ = 5; -alert(obj.__proto__); // [object Object], didn't work as intended -``` - -As we see from the code, the assignment to a primitive `5` is ignored. - -That can become a source of bugs and even vulnerabilities if we intend to store arbitrary key-value pairs in an object, and allow a visitor to specify the keys. - -In that case the visitor may choose "__proto__" as the key, and the assignment logic will be ruined (as shown above). - -There is a way to make objects treat `__proto__` as a regular property, which we'll cover later, but first we need to know more about objects. -There's also another data structure [Map](info:map-set-weakmap-weakset), that we'll learn in the chapter <info:map-set-weakmap-weakset>, which supports arbitrary keys. -```` - - ## Property value shorthand -In real code we often use existing variables as values for property names. +In real code, we often use existing variables as values for property names. For instance: @@ -239,7 +215,7 @@ For instance: function makeUser(name, age) { return { name: name, - age: age + age: age, // ...other properties }; } @@ -257,7 +233,7 @@ function makeUser(name, age) { *!* return { name, // same as name: name - age // same as age: age + age, // same as age: age // ... }; */!* @@ -273,9 +249,57 @@ let user = { }; ``` -## Existence check -A notable objects feature is that it's possible to access any property. There will be no error if the property doesn't exist! Accessing a non-existing property just returns `undefined`. It provides a very common way to test whether the property exists -- to get it and compare vs undefined: +## Property names limitations + +As we already know, a variable cannot have a name equal to one of the language-reserved words like "for", "let", "return" etc. + +But for an object property, there's no such restriction: + +```js run +// these properties are all right +let obj = { + for: 1, + let: 2, + return: 3 +}; + +alert( obj.for + obj.let + obj.return ); // 6 +``` + +In short, there are no limitations on property names. They can be any strings or symbols (a special type for identifiers, to be covered later). + +Other types are automatically converted to strings. + +For instance, a number `0` becomes a string `"0"` when used as a property key: + +```js run +let obj = { + 0: "test" // same as "0": "test" +}; + +// both alerts access the same property (the number 0 is converted to string "0") +alert( obj["0"] ); // test +alert( obj[0] ); // test (same property) +``` + +There's a minor gotcha with a special property named `__proto__`. We can't set it to a non-object value: + +```js run +let obj = {}; +obj.__proto__ = 5; // assign a number +alert(obj.__proto__); // [object Object] - the value is an object, didn't work as intended +``` + +As we see from the code, the assignment to a primitive `5` is ignored. + +We'll cover the special nature of `__proto__` in [subsequent chapters](info:prototype-inheritance), and suggest the [ways to fix](info:prototype-methods) such behavior. + +## Property existence test, "in" operator + +A notable feature of objects in JavaScript, compared to many other languages, is that it's possible to access any property. There will be no error if the property doesn't exist! + +Reading a non-existing property just returns `undefined`. So we can easily test whether the property exists: ```js run let user = {}; @@ -283,7 +307,7 @@ let user = {}; alert( user.noSuchProperty === undefined ); // true means "no such property" ``` -There also exists a special operator `"in"` to check for the existence of a property. +There's also a special operator `"in"` for that. The syntax is: ```js @@ -301,17 +325,18 @@ alert( "blabla" in user ); // false, user.blabla doesn't exist Please note that on the left side of `in` there must be a *property name*. That's usually a quoted string. -If we omit quotes, that would mean a variable containing the actual name will be tested. For instance: +If we omit quotes, that means a variable should contain the actual name to be tested. For instance: ```js run let user = { age: 30 }; let key = "age"; -alert( *!*key*/!* in user ); // true, takes the name from key and checks for such property +alert( *!*key*/!* in user ); // true, property "age" exists ``` -````smart header="Using \"in\" for properties that store `undefined`" -Usually, the strict comparison `"=== undefined"` check works fine. But there's a special case when it fails, but `"in"` works correctly. +Why does the `in` operator exist? Isn't it enough to compare against `undefined`? + +Well, most of the time the comparison with `undefined` works fine. But there's a special case when it fails, but `"in"` works correctly. It's when an object property exists, but stores `undefined`: @@ -325,14 +350,12 @@ alert( obj.test ); // it's undefined, so - no such property? alert( "test" in obj ); // true, the property does exist! ``` - In the code above, the property `obj.test` technically exists. So the `in` operator works right. -Situations like this happen very rarely, because `undefined` is usually not assigned. We mostly use `null` for "unknown" or "empty" values. So the `in` operator is an exotic guest in the code. -```` +Situations like this happen very rarely, because `undefined` should not be explicitly assigned. We mostly use `null` for "unknown" or "empty" values. So the `in` operator is an exotic guest in the code. -## The "for..in" loop +## The "for..in" loop [#forin] To walk over all keys of an object, there exists a special form of the loop: `for..in`. This is a completely different thing from the `for(;;)` construct that we studied before. @@ -365,7 +388,6 @@ Note that all "for" constructs allow us to declare the looping variable inside t Also, we could use another variable name here instead of `key`. For instance, `"for (let prop in obj)"` is also widely used. - ### Ordered like an object Are objects ordered? In other words, if we loop over an object, do we get all properties in the same order they were added? Can we rely on this? @@ -390,7 +412,7 @@ for (let code in codes) { */!* ``` -The object may be used to suggest a list of options to the user. If we're making a site mainly for German audience then we probably want `49` to be the first. +The object may be used to suggest a list of options to the user. If we're making a site mainly for a German audience then we probably want `49` to be the first. But if we run the code, we see a totally different picture: @@ -402,9 +424,10 @@ The phone codes go in the ascending sorted order, because they are integers. So ````smart header="Integer properties? What's that?" The "integer property" term here means a string that can be converted to-and-from an integer without a change. -So, "49" is an integer property name, because when it's transformed to an integer number and back, it's still the same. But "+49" and "1.2" are not: +So, `"49"` is an integer property name, because when it's transformed to an integer number and back, it's still the same. But `"+49"` and `"1.2"` are not: ```js run +// Number(...) explicitly converts to a number // Math.trunc is a built-in function that removes the decimal part alert( String(Math.trunc(Number("49"))) ); // "49", same, integer property alert( String(Math.trunc(Number("+49"))) ); // "49", not same "+49" ⇒ not integer property @@ -449,6 +472,7 @@ for (let code in codes) { Now it works as intended. +<<<<<<< HEAD ## Copying by reference One of the fundamental differences of objects vs primitives is that they are stored and copied "by reference". @@ -705,6 +729,8 @@ There's a standard algorithm for deep cloning that handles the case above and mo +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Summary Objects are associative arrays with several special features. @@ -715,17 +741,13 @@ They store properties (key-value pairs), where: To access a property, we can use: - The dot notation: `obj.property`. -- Square brackets notation `obj["property"]`. Square brackets allow to take the key from a variable, like `obj[varWithKey]`. +- Square brackets notation `obj["property"]`. Square brackets allow taking the key from a variable, like `obj[varWithKey]`. Additional operators: - To delete a property: `delete obj.prop`. - To check if a property with the given key exists: `"key" in obj`. - To iterate over an object: `for (let key in obj)` loop. -Objects are assigned and copied by reference. In other words, a variable stores not the "object value", but a "reference" (address in memory) for the value. So copying such a variable or passing it as a function argument copies that reference, not the object. All operations via copied references (like adding/removing properties) are performed on the same single object. - -To make a "real copy" (a clone) we can use `Object.assign` or [_.cloneDeep(obj)](https://lodash.com/docs#cloneDeep). - What we've studied in this chapter is called a "plain object", or just `Object`. There are many other kinds of objects in JavaScript: diff --git a/1-js/04-object-basics/02-garbage-collection/memory-user-john.svg b/1-js/04-object-basics/02-garbage-collection/memory-user-john.svg deleted file mode 100644 index 45ce86ed2..000000000 --- a/1-js/04-object-basics/02-garbage-collection/memory-user-john.svg +++ /dev/null @@ -1,25 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> -<svg width="144px" height="150px" viewBox="0 0 144 150" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> - <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> - <title>memory-user-john.svg</title> - <desc>Created with sketchtool.</desc> - <g id="garbage-collection" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> - <g id="memory-user-john.svg"> - <text id="user" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> - <tspan x="31" y="69">user</tspan> - </text> - <rect id="Rectangle-4" stroke="#BCA68E" stroke-width="2" fill="#FFF9EB" x="16" y="95" width="118" height="48"></rect> - <text id="name:-"John"" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" fill="#8A704D"> - <tspan x="32" y="132">name: "John"</tspan> - </text> - <text id="Object-3" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> - <tspan x="46" y="112">Object</tspan> - </text> - <path id="Line" d="M73.5,67.5 L73.5,49.5 L75.5,49.5 L75.5,67.5 L81.5,67.5 L74.5,81.5 L67.5,67.5 L73.5,67.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - <rect id="Rectangle-1" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB" x="27" y="12" width="93" height="26"></rect> - <text id="<global>" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> - <tspan x="43" y="29"><global></tspan> - </text> - </g> - </g> -</svg> \ No newline at end of file diff --git a/1-js/04-object-basics/02-object-copy/article.md b/1-js/04-object-basics/02-object-copy/article.md new file mode 100644 index 000000000..e80f748ab --- /dev/null +++ b/1-js/04-object-basics/02-object-copy/article.md @@ -0,0 +1,325 @@ +# Object references and copying + +One of the fundamental differences of objects versus primitives is that objects are stored and copied "by reference", whereas primitive values: strings, numbers, booleans, etc -- are always copied "as a whole value". + +That's easy to understand if we look a bit under the hood of what happens when we copy a value. + +Let's start with a primitive, such as a string. + +Here we put a copy of `message` into `phrase`: + +```js +let message = "Hello!"; +let phrase = message; +``` + +As a result we have two independent variables, each one storing the string `"Hello!"`. + + + +Quite an obvious result, right? + +Objects are not like that. + +**A variable assigned to an object stores not the object itself, but its "address in memory" -- in other words "a reference" to it.** + +Let's look at an example of such a variable: + +```js +let user = { + name: "John" +}; +``` + +And here's how it's actually stored in memory: + + + +The object is stored somewhere in memory (at the right of the picture), while the `user` variable (at the left) has a "reference" to it. + +We may think of an object variable, such as `user`, like a sheet of paper with the address of the object on it. + +When we perform actions with the object, e.g. take a property `user.name`, the JavaScript engine looks at what's at that address and performs the operation on the actual object. + +Now here's why it's important. + +**When an object variable is copied, the reference is copied, but the object itself is not duplicated.** + +For instance: + +```js no-beautify +let user = { name: "John" }; + +let admin = user; // copy the reference +``` + +Now we have two variables, each storing a reference to the same object: + + + +As you can see, there's still one object, but now with two variables that reference it. + +We can use either variable to access the object and modify its contents: + +```js run +let user = { name: 'John' }; + +let admin = user; + +*!* +admin.name = 'Pete'; // changed by the "admin" reference +*/!* + +alert(*!*user.name*/!*); // 'Pete', changes are seen from the "user" reference +``` + +It's as if we had a cabinet with two keys and used one of them (`admin`) to get into it and make changes. Then, if we later use another key (`user`), we are still opening the same cabinet and can access the changed contents. + +## Comparison by reference + +Two objects are equal only if they are the same object. + +For instance, here `a` and `b` reference the same object, thus they are equal: + +```js run +let a = {}; +let b = a; // copy the reference + +alert( a == b ); // true, both variables reference the same object +alert( a === b ); // true +``` + +And here two independent objects are not equal, even though they look alike (both are empty): + +```js run +let a = {}; +let b = {}; // two independent objects + +alert( a == b ); // false +``` + +For comparisons like `obj1 > obj2` or for a comparison against a primitive `obj == 5`, objects are converted to primitives. We'll study how object conversions work very soon, but to tell the truth, such comparisons are needed very rarely -- usually they appear as a result of a programming mistake. + +````smart header="Const objects can be modified" +An important side effect of storing objects as references is that an object declared as `const` *can* be modified. + +For instance: + +```js run +const user = { + name: "John" +}; + +*!* +user.name = "Pete"; // (*) +*/!* + +alert(user.name); // Pete +``` + +It might seem that the line `(*)` would cause an error, but it does not. The value of `user` is constant, it must always reference the same object, but properties of that object are free to change. + +In other words, the `const user` gives an error only if we try to set `user=...` as a whole. + +That said, if we really need to make constant object properties, it's also possible, but using totally different methods. We'll mention that in the chapter <info:property-descriptors>. +```` + +## Cloning and merging, Object.assign [#cloning-and-merging-object-assign] + +So, copying an object variable creates one more reference to the same object. + +But what if we need to duplicate an object? + +We can create a new object and replicate the structure of the existing one, by iterating over its properties and copying them on the primitive level. + +Like this: + +```js run +let user = { + name: "John", + age: 30 +}; + +*!* +let clone = {}; // the new empty object + +// let's copy all user properties into it +for (let key in user) { + clone[key] = user[key]; +} +*/!* + +// now clone is a fully independent object with the same content +clone.name = "Pete"; // changed the data in it + +alert( user.name ); // still John in the original object +``` + +We can also use the method [Object.assign](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/assign). + +The syntax is: + +```js +Object.assign(dest, ...sources) +``` + +- The first argument `dest` is a target object. +- Further arguments is a list of source objects. + +It copies the properties of all source objects into the target `dest`, and then returns it as the result. + +For example, we have `user` object, let's add a couple of permissions to it: + +```js run +let user = { name: "John" }; + +let permissions1 = { canView: true }; +let permissions2 = { canEdit: true }; + +*!* +// copies all properties from permissions1 and permissions2 into user +Object.assign(user, permissions1, permissions2); +*/!* + +// now user = { name: "John", canView: true, canEdit: true } +alert(user.name); // John +alert(user.canView); // true +alert(user.canEdit); // true +``` + +If the copied property name already exists, it gets overwritten: + +```js run +let user = { name: "John" }; + +Object.assign(user, { name: "Pete" }); + +alert(user.name); // now user = { name: "Pete" } +``` + +We also can use `Object.assign` to perform a simple object cloning: + +```js run +let user = { + name: "John", + age: 30 +}; + +*!* +let clone = Object.assign({}, user); +*/!* + +alert(clone.name); // John +alert(clone.age); // 30 +``` + +Here it copies all properties of `user` into the empty object and returns it. + +There are also other methods of cloning an object, e.g. using the [spread syntax](info:rest-parameters-spread) `clone = {...user}`, covered later in the tutorial. + +## Nested cloning + +Until now we assumed that all properties of `user` are primitive. But properties can be references to other objects. + +Like this: +```js run +let user = { + name: "John", + sizes: { + height: 182, + width: 50 + } +}; + +alert( user.sizes.height ); // 182 +``` + +Now it's not enough to copy `clone.sizes = user.sizes`, because `user.sizes` is an object, and will be copied by reference, so `clone` and `user` will share the same sizes: + +```js run +let user = { + name: "John", + sizes: { + height: 182, + width: 50 + } +}; + +let clone = Object.assign({}, user); + +alert( user.sizes === clone.sizes ); // true, same object + +// user and clone share sizes +user.sizes.width = 60; // change a property from one place +alert(clone.sizes.width); // 60, get the result from the other one +``` + +To fix that and make `user` and `clone` truly separate objects, we should use a cloning loop that examines each value of `user[key]` and, if it's an object, then replicate its structure as well. That is called a "deep cloning" or "structured cloning". There's [structuredClone](https://developer.mozilla.org/en-US/docs/Web/API/structuredClone) method that implements deep cloning. + + +### structuredClone + +The call `structuredClone(object)` clones the `object` with all nested properties. + +Here's how we can use it in our example: + +```js run +let user = { + name: "John", + sizes: { + height: 182, + width: 50 + } +}; + +*!* +let clone = structuredClone(user); +*/!* + +alert( user.sizes === clone.sizes ); // false, different objects + +// user and clone are totally unrelated now +user.sizes.width = 60; // change a property from one place +alert(clone.sizes.width); // 50, not related +``` + +The `structuredClone` method can clone most data types, such as objects, arrays, primitive values. + +It also supports circular references, when an object property references the object itself (directly or via a chain or references). + +For instance: + +```js run +let user = {}; +// let's create a circular reference: +// user.me references the user itself +user.me = user; + +let clone = structuredClone(user); +alert(clone.me === clone); // true +``` + +As you can see, `clone.me` references the `clone`, not the `user`! So the circular reference was cloned correctly as well. + +Although, there are cases when `structuredClone` fails. + +For instance, when an object has a function property: + +```js run +// error +structuredClone({ + f: function() {} +}); +``` + +Function properties aren't supported. + +To handle such complex cases we may need to use a combination of cloning methods, write custom code or, to not reinvent the wheel, take an existing implementation, for instance [_.cloneDeep(obj)](https://lodash.com/docs#cloneDeep) from the JavaScript library [lodash](https://lodash.com). + +## Summary + +Objects are assigned and copied by reference. In other words, a variable stores not the "object value", but a "reference" (address in memory) for the value. So copying such a variable or passing it as a function argument copies that reference, not the object itself. + +All operations via copied references (like adding/removing properties) are performed on the same single object. + +To make a "real copy" (a clone) we can use `Object.assign` for the so-called "shallow copy" (nested objects are copied by reference) or a "deep cloning" function `structuredClone` or use a custom cloning implementation, such as [_.cloneDeep(obj)](https://lodash.com/docs#cloneDeep). diff --git a/1-js/04-object-basics/02-object-copy/variable-contains-reference.svg b/1-js/04-object-basics/02-object-copy/variable-contains-reference.svg new file mode 100644 index 000000000..267f04578 --- /dev/null +++ b/1-js/04-object-basics/02-object-copy/variable-contains-reference.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="370" height="249" viewBox="0 0 370 249"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="variable-contains-reference.svg"><g id="noun_1211_cc" fill="#DBAF88" transform="translate(12 119)"><path id="Shape" d="M17 28.196h112.558v42.95c0 .373-.079.862-.279 1.294-.2.433-16.574 35.56-16.574 35.56V53.64l16.854-25.444L148 0H35.44L17 28.196zM17 57V29L2 57"/><path id="Shape" d="M0 59v54.73c0 3.42 1.484 5.27 4.387 5.27h100.086c3.122 0 5.527-2.548 5.527-3.476V59H0z"/></g><text id="user" fill="#FFF" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="47" y="215">user</tspan></text><path id="Rectangle-4" fill="#FFF" d="M116 153h44v6h-44z"/><path id="Shape-3" fill="#C06334" d="M123.854 154.322l13.779-13.476c1.613-1.274 5.185-4.427 3.572-5.702-1.613-1.275-2.664-.472-4.275.803l-18.499 18.375c-1.613 1.275-1.613 2.845 0 4.12 0 0 21.483 20.067 22.774 20.067 3.005 0 2.677-1.821 1.064-3.096l-18.415-17.826H201.028c2.281 0 3.75.279 3.75-1.524 0-1.804-1.469-1.741-3.75-1.741h-77.174z" transform="matrix(-1 0 0 1 322 0)"/><g id="Rectangle-3-+-Shape" transform="translate(58 135)"><path id="Rectangle-3" fill="#FFF" d="M0 0h47v37H0z"/><path id="Shape" fill="#C06334" d="M44 33.772H4V4h40v29.772zM8.733 31.018h30.533L29.22 20.868l-1.902 1.889c-.824.812-2.003 1.277-3.242 1.277l-.021-.001c-1.244-.005-2.427-.48-3.246-1.304l-1.934-1.933-10.14 10.222zM6.791 8.726V29.04l10.122-10.202L6.79 8.726zm24.395 10.187L41.209 29.04V8.954l-10.023 9.96zM8.74 6.755l14.057 14.042c.304.305.766.48 1.27.482h.008c.496 0 .968-.173 1.264-.467L39.49 6.755H8.739z"/></g><path id="Rectangle-4-Copy" fill="#DBAF88" stroke="#DBAF88" stroke-width="4" d="M339.063 117l16.667 20H225.27l16.667-20h97.126z" opacity=".5"/><g id="Group-2" transform="translate(271 10)"><g id="Group"><path id="Rectangle-7" fill="#FFF" d="M0 0h32v124H0z"/><path id="Combined-Shape" fill="#DBAF88" d="M32 0v124H0V0h32zM16.5 100a7.5 7.5 0 100 15 7.5 7.5 0 000-15zM28 5H4v87h24V5z"/></g><text id="name" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold" transform="rotate(-90 15.5 47.5)"><tspan x="-3.7" y="52.5">name</tspan></text></g><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" d="M223 131h135v50H223z"/><path id="Rectangle-8" stroke="#DBAF88" stroke-width="3" d="M278.5 151.5h25v10h-25z"/></g></g></svg> \ No newline at end of file diff --git a/1-js/04-object-basics/02-object-copy/variable-copy-reference.svg b/1-js/04-object-basics/02-object-copy/variable-copy-reference.svg new file mode 100644 index 000000000..a847fb200 --- /dev/null +++ b/1-js/04-object-basics/02-object-copy/variable-copy-reference.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="599" height="260" viewBox="0 0 599 260"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="variable-copy-reference.svg"><g id="Group" transform="translate(11 125)"><g id="noun_1211_cc" fill="#DBAF88"><path id="Shape" d="M17 28.196h112.558v42.95c0 .373-.079.862-.279 1.294-.2.433-16.574 35.56-16.574 35.56V53.64l16.854-25.444L148 0H35.44L17 28.196zM17 57V29L2 57"/><path id="Shape" d="M0 59v54.73c0 3.42 1.484 5.27 4.387 5.27h100.086c3.122 0 5.527-2.548 5.527-3.476V59H0z"/></g><text id="user" fill="#FFF" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="35" y="96">user</tspan></text><g id="Rectangle-4-+-Shape-3" transform="translate(104 15)"><path id="Rectangle-4" fill="#FFF" d="M0 19h44v6H0z"/><path id="Shape-3" fill="#C06334" d="M7.854 20.322L21.633 6.846c1.613-1.274 5.185-4.427 3.572-5.702-1.613-1.275-2.664-.472-4.275.803L2.43 20.322c-1.613 1.275-1.613 2.845 0 4.12 0 0 21.483 20.067 22.774 20.067 3.005 0 2.677-1.821 1.064-3.096L7.854 23.587H85.028c2.281 0 3.75.279 3.75-1.524 0-1.804-1.469-1.741-3.75-1.741H7.854z" transform="matrix(-1 0 0 1 90 0)"/></g><g id="Rectangle-3-+-Shape" transform="translate(46 16)"><path id="Rectangle-3" fill="#FFF" d="M0 0h47v37H0z"/><path id="Shape" fill="#C06334" d="M44 33.772H4V4h40v29.772zM8.733 31.018h30.533L29.22 20.868l-1.902 1.889c-.824.812-2.003 1.277-3.242 1.277l-.021-.001c-1.244-.005-2.427-.48-3.246-1.304l-1.934-1.933-10.14 10.222zM6.791 8.726V29.04l10.122-10.202L6.79 8.726zm24.395 10.187L41.209 29.04V8.954l-10.023 9.96zM8.74 6.755l14.057 14.042c.304.305.766.48 1.27.482h.008c.496 0 .968-.173 1.264-.467L39.49 6.755H8.739z"/></g></g><g id="noun_1211_cc" fill="#DBAF88" transform="translate(418 125)"><path id="Shape" d="M17 28.196h112.558v42.95c0 .373-.079.862-.279 1.294-.2.433-16.574 35.56-16.574 35.56V53.64l16.854-25.444L148 0H35.44L17 28.196zM17 57V29L2 57"/><path id="Shape" d="M0 59v54.73c0 3.42 1.484 5.27 4.387 5.27h100.086c3.122 0 5.527-2.548 5.527-3.476V59H0z"/></g><text id="admin" fill="#FFF" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="446" y="221">admin</tspan></text><g id="Rectangle-4-+-Shape-3" transform="matrix(-1 0 0 1 451 140)"><path id="Rectangle-4" fill="#FFF" d="M0 19h44v6H0z"/><path id="Shape-3" fill="#C06334" d="M7.854 20.322L21.633 6.846c1.613-1.274 5.185-4.427 3.572-5.702-1.613-1.275-2.664-.472-4.275.803L2.43 20.322c-1.613 1.275-1.613 2.845 0 4.12 0 0 21.483 20.067 22.774 20.067 3.005 0 2.677-1.821 1.064-3.096L7.854 23.587H85.028c2.281 0 3.75.279 3.75-1.524 0-1.804-1.469-1.741-3.75-1.741H7.854z" transform="matrix(-1 0 0 1 90 0)"/></g><g id="Rectangle-3-+-Shape" transform="translate(464 141)"><path id="Rectangle-3" fill="#FFF" d="M0 0h47v37H0z"/><path id="Shape" fill="#C06334" d="M44 33.772H4V4h40v29.772zM8.733 31.018h30.533L29.22 20.868l-1.902 1.889c-.824.812-2.003 1.277-3.242 1.277l-.021-.001c-1.244-.005-2.427-.48-3.246-1.304l-1.934-1.933-10.14 10.222zM6.791 8.726V29.04l10.122-10.202L6.79 8.726zm24.395 10.187L41.209 29.04V8.954l-10.023 9.96zM8.74 6.755l14.057 14.042c.304.305.766.48 1.27.482h.008c.496 0 .968-.173 1.264-.467L39.49 6.755H8.739z"/></g><path id="Rectangle-4-Copy" fill="#DBAF88" stroke="#DBAF88" stroke-width="4" d="M333.063 125l16.667 20H219.27l16.667-20h97.126z" opacity=".5"/><g id="Group-2" transform="translate(265 18)"><g id="Group"><path id="Rectangle-7" fill="#FFF" d="M0 0h32v124H0z"/><path id="Combined-Shape" fill="#DBAF88" d="M32 0v124H0V0h32zM16.5 100a7.5 7.5 0 100 15 7.5 7.5 0 000-15zM28 5H4v87h24V5z"/></g><text id="name" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold" transform="rotate(-90 15.5 47.5)"><tspan x="-3.7" y="52.5">name</tspan></text></g><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" d="M217 139h135v50H217z"/><path id="Rectangle-8" stroke="#DBAF88" stroke-width="3" d="M272.5 159.5h25v10h-25z"/></g></g></svg> \ No newline at end of file diff --git a/1-js/04-object-basics/02-object-copy/variable-copy-value.svg b/1-js/04-object-basics/02-object-copy/variable-copy-value.svg new file mode 100644 index 000000000..0d6ca67bc --- /dev/null +++ b/1-js/04-object-basics/02-object-copy/variable-copy-value.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="359" height="143" viewBox="0 0 359 143"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="variable-copy-value.svg"><g id="noun_1211_cc-+-Message" transform="translate(11 6)"><g id="noun_1211_cc"><path id="Shape" fill="#DBAF88" d="M17 37.196h112.558v42.95c0 .373-.079.862-.279 1.294-.2.433-16.574 35.56-16.574 35.56V62.64l16.854-25.444L148 9H35.44L17 37.196zM17 66V38L2 66"/><g id="Rectangle-5-+-"World!"" transform="translate(15)"><path id="Rectangle-5" fill="#FBF2EC" stroke="#AF6E24" stroke-width="2" d="M18.861 1.809l53.14 56.985L55.14 74.52 1.999 17.533 18.86 1.81z"/><text id=""Hello!"" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold" transform="rotate(47 38.202 38.946)"><tspan x="2.822" y="43.482">"Hello!"</tspan></text></g><path id="Shape" fill="#DBAF88" d="M0 68v54.73c0 3.42 1.484 5.27 4.387 5.27h100.086c3.122 0 5.527-2.548 5.527-3.476V68H0z"/></g><text id="message" fill="#FFF" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="17" y="105">message</tspan></text></g><g id="Group" transform="translate(210 6)"><path id="Shape" fill="#DBAF88" d="M17 37.196h113.417v42.95c0 .373-.08.862-.28 1.294-.202.433-16.702 35.56-16.702 35.56V62.64l16.983-25.444L149 9H35.582L17 37.196z"/><path id="Shape" fill="#DBAF88" d="M18 66V38L2 66"/><g id="Rectangle-5-+-"World!"" transform="translate(15)"><path id="Rectangle-5" fill="#FBF2EC" stroke="#AF6E24" stroke-width="2" d="M19.117 1.8l53.867 56.994-17.101 15.734L2.016 17.534 19.117 1.799z"/><text id=""Hello!"" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold" transform="rotate(47 38.162 37.693)"><tspan x="2.782" y="42.23">"Hello!"</tspan></text></g><path id="Shape" fill="#DBAF88" d="M0 68v54.73c0 3.42 1.497 5.27 4.427 5.27h100.996c3.15 0 5.577-2.548 5.577-3.476V68H0z"/><text id="phrase" fill="#FFF" font-family="OpenSans-Bold, Open Sans" font-size="18" font-weight="bold"><tspan x="25" y="105">phrase</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/04-object-basics/02-garbage-collection/article.md b/1-js/04-object-basics/03-garbage-collection/article.md similarity index 66% rename from 1-js/04-object-basics/02-garbage-collection/article.md rename to 1-js/04-object-basics/03-garbage-collection/article.md index 3b385afd2..1b576d629 100644 --- a/1-js/04-object-basics/02-garbage-collection/article.md +++ b/1-js/04-object-basics/03-garbage-collection/article.md @@ -14,8 +14,8 @@ Simply put, "reachable" values are those that are accessible or usable somehow. For instance: - - Local variables and parameters of the current function. - - Variables and parameters for other functions on the current chain of nested calls. + - The currently executing function, its local variables and parameters. + - Other functions on the current chain of nested calls, their local variables and parameters. - Global variables. - (there are some other, internal ones as well) @@ -23,7 +23,7 @@ Simply put, "reachable" values are those that are accessible or usable somehow. 2. Any other value is considered reachable if it's reachable from a root by a reference or by a chain of references. - For instance, if there's an object in a local variable, and that object has a property referencing another object, that object is considered reachable. And those that it references are also reachable. Detailed examples to follow. + For instance, if there's an object in a global variable, and that object has a property referencing another object, *that* object is considered reachable. And those that it references are also reachable. Detailed examples to follow. There's a background process in the JavaScript engine that is called [garbage collector](https://en.wikipedia.org/wiki/Garbage_collection_(computer_science)). It monitors all objects and removes those that have become unreachable. @@ -74,7 +74,7 @@ Now if we do the same: user = null; ``` -...Then the object is still reachable via `admin` global variable, so it's in memory. If we overwrite `admin` too, then it can be removed. +...Then the object is still reachable via `admin` global variable, so it must stay in memory. If we overwrite `admin` too, then it can be removed. ## Interlinked objects @@ -156,7 +156,7 @@ The following "garbage collection" steps are regularly performed: - The garbage collector takes roots and "marks" (remembers) them. - Then it visits and "marks" all references from them. - Then it visits marked objects and marks *their* references. All visited objects are remembered, so as not to visit the same object twice in the future. -- ...And so on until there are unvisited references (reachable from the roots). +- ...And so on until every reachable (from the roots) references are visited. - All objects except marked ones are removed. For instance, let our object structure look like this: @@ -169,11 +169,11 @@ The first step marks the roots:  -Then their references are marked: +Then we follow their references and mark referenced objects:  -...And their references, while possible: +...And continue to follow further references, while possible:  @@ -181,17 +181,17 @@ Now the objects that could not be visited in the process are considered unreacha  -That's the concept of how garbage collection works. +We can also imagine the process as spilling a huge bucket of paint from the roots, that flows through all references and marks all reachable objects. The unmarked ones are then removed. -JavaScript engines apply many optimizations to make it run faster and not affect the execution. +That's the concept of how garbage collection works. JavaScript engines apply many optimizations to make it run faster and not introduce any delays into the code execution. Some of the optimizations: -- **Generational collection** -- objects are split into two sets: "new ones" and "old ones". Many objects appear, do their job and die fast, they can be cleaned up aggressively. Those that survive for long enough, become "old" and are examined less often. -- **Incremental collection** -- if there are many objects, and we try to walk and mark the whole object set at once, it may take some time and introduce visible delays in the execution. So the engine tries to split the garbage collection into pieces. Then the pieces are executed one by one, separately. That requires some extra bookkeeping between them to track changes, but we have many tiny delays instead of a big one. +- **Generational collection** -- objects are split into two sets: "new ones" and "old ones". In typical code, many objects have a short life span: they appear, do their job and die fast, so it makes sense to track new objects and clear the memory from them if that's the case. Those that survive for long enough, become "old" and are examined less often. +- **Incremental collection** -- if there are many objects, and we try to walk and mark the whole object set at once, it may take some time and introduce visible delays in the execution. So the engine splits the whole set of existing objects into multiple parts. And then clear these parts one after another. There are many small garbage collections instead of a total one. That requires some extra bookkeeping between them to track changes, but we get many tiny delays instead of a big one. - **Idle-time collection** -- the garbage collector tries to run only while the CPU is idle, to reduce the possible effect on the execution. -There are other optimizations and flavours of garbage collection algorithms. As much as I'd like to describe them here, I have to hold off, because different engines implement different tweaks and techniques. And, what's even more important, things change as engines develop, so going deeper "in advance", without a real need is probably not worth that. Unless, of course, it is a matter of pure interest, then there will be some links for you below. +There exist other optimizations and flavours of garbage collection algorithms. As much as I'd like to describe them here, I have to hold off, because different engines implement different tweaks and techniques. And, what's even more important, things change as engines develop, so studying deeper "in advance", without a real need is probably not worth that. Unless, of course, it is a matter of pure interest, then there will be some links for you below. ## Summary @@ -199,14 +199,14 @@ The main things to know: - Garbage collection is performed automatically. We cannot force or prevent it. - Objects are retained in memory while they are reachable. -- Being referenced is not the same as being reachable (from a root): a pack of interlinked objects can become unreachable as a whole. +- Being referenced is not the same as being reachable (from a root): a pack of interlinked objects can become unreachable as a whole, as we've seen in the example above. Modern engines implement advanced algorithms of garbage collection. A general book "The Garbage Collection Handbook: The Art of Automatic Memory Management" (R. Jones et al) covers some of them. -If you are familiar with low-level programming, the more detailed information about V8 garbage collector is in the article [A tour of V8: Garbage Collection](http://jayconrod.com/posts/55/a-tour-of-v8-garbage-collection). +If you are familiar with low-level programming, more detailed information about V8's garbage collector is in the article [A tour of V8: Garbage Collection](https://jayconrod.com/posts/55/a-tour-of-v8-garbage-collection). -[V8 blog](http://v8project.blogspot.com/) also publishes articles about changes in memory management from time to time. Naturally, to learn the garbage collection, you'd better prepare by learning about V8 internals in general and read the blog of [Vyacheslav Egorov](http://mrale.ph) who worked as one of V8 engineers. I'm saying: "V8", because it is best covered with articles in the internet. For other engines, many approaches are similar, but garbage collection differs in many aspects. +The [V8 blog](https://v8.dev/) also publishes articles about changes in memory management from time to time. Naturally, to learn more about garbage collection, you'd better prepare by learning about V8 internals in general and read the blog of [Vyacheslav Egorov](https://mrale.ph) who worked as one of the V8 engineers. I'm saying: "V8", because it is best covered by articles on the internet. For other engines, many approaches are similar, but garbage collection differs in many aspects. -In-depth knowledge of engines is good when you need low-level optimizations. It would be wise to plan that as the next step after you're familiar with the language. +In-depth knowledge of engines is good when you need low-level optimizations. It would be wise to plan that as the next step after you're familiar with the language. diff --git a/1-js/04-object-basics/02-garbage-collection/family-delete-refs.svg b/1-js/04-object-basics/03-garbage-collection/family-delete-refs.svg similarity index 55% rename from 1-js/04-object-basics/02-garbage-collection/family-delete-refs.svg rename to 1-js/04-object-basics/03-garbage-collection/family-delete-refs.svg index a67ffc3cc..1a98116e3 100644 --- a/1-js/04-object-basics/02-garbage-collection/family-delete-refs.svg +++ b/1-js/04-object-basics/03-garbage-collection/family-delete-refs.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/family-delete-refs.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="337px" height="204px" viewBox="0 0 337 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -53,4 +54,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="337" height="204" viewBox="0 0 337 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="family-delete-refs.svg"><path id="Line" fill="#7E7C7B" fill-rule="nonzero" d="M147.5 183.5v6h46v2h-46v6l-14-7 14-7z" opacity=".6"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M88 13h151v26H88z"/><text id="<global-variable>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="92" y="30"><global variable></tspan></text><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M123 80h78v26h-78z"/><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="139" y="96">Object</tspan></text><path id="Rectangle-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M5 148h118v48H5z"/><text id="Object-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="39" y="165">Object</tspan></text><text id="wife" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="147" y="148">wife</tspan></text><text id="family" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="106" y="63">family</tspan></text><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="19" y="185">name: "John"</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M214 148h118v48H214z"/><text id="name:-"Ann"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="230" y="185">name: "Ann"</tspan></text><text id="mother" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="242" y="127">mother</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="244" y="165">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M164.5 43.5v18h6l-7 14-7-14h6v-18h2z"/><path id="Line-2" fill="#7E7C7B" fill-rule="nonzero" d="M114.75 111.108l1.142 1.642-.82.57-34.508 24.005 3.426 4.926L68.5 144.5l7.495-13.741 3.426 4.924 34.508-24.004.82-.571z" opacity=".6"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M180.5 149.5l14 7-14 7v-6h-48v-2h48v-6zM208.236 111.11l.827.563 35.427 24.121 3.377-4.96L255.5 144.5l-15.512-2.093 3.377-4.96-35.428-24.12-.826-.563 1.125-1.653z"/><path id="Line-Copy" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M100.5 118.5l-11 14"/><path id="Line-Copy-2" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M89.5 118.5l11 14"/><path id="Line-Copy-4" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M172.5 183.5l-11 14"/><path id="Line-Copy-3" stroke="#C06334" stroke-linecap="square" stroke-width="2" d="M161.5 183.5l11 14"/><text id="father" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="35" y="127">father</tspan></text><text id="husband" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="136" y="181">husband</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/family-delete-refs.svg diff --git a/1-js/04-object-basics/02-garbage-collection/family-no-family.svg b/1-js/04-object-basics/03-garbage-collection/family-no-family.svg similarity index 60% rename from 1-js/04-object-basics/02-garbage-collection/family-no-family.svg rename to 1-js/04-object-basics/03-garbage-collection/family-no-family.svg index df8601215..eb023f4ee 100644 --- a/1-js/04-object-basics/02-garbage-collection/family-no-family.svg +++ b/1-js/04-object-basics/03-garbage-collection/family-no-family.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/family-no-family.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="420px" height="279px" viewBox="0 0 420 279" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -54,4 +55,7 @@ </g> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="420" height="279" viewBox="0 0 420 279"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="family-no-family.svg"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M211.5 228.5v6h46v2h-46v6l-14-7 14-7z"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M153 25h153v48H153z"/><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="199" y="43"><global></tspan></text><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M187 123h78v26h-78z"/><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="203" y="139">Object</tspan></text><path id="Rectangle-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M69 193h118v48H69z"/><text id="Object-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="103" y="210">Object</tspan></text><text id="father" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="99" y="170">father</tspan></text><text id="wife" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="211" y="193">wife</tspan></text><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="83" y="230">name: "John"</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M278 193h118v48H278z"/><text id="name:-"Ann"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="294" y="230">name: "Ann"</tspan></text><text id="mother" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="306" y="170">mother</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="308" y="210">Object</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M178.75 154.108l1.142 1.642-.82.57-34.508 24.005 3.426 4.926-15.49 2.249 7.495-13.741 3.426 4.924 34.508-24.004.82-.571z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M244.5 194.5l14 7-14 7v-6h-48v-2h48v-6z"/><text id="husband" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="200" y="226">husband</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M272.236 154.11l.827.563 35.427 24.121 3.377-4.96L319.5 187.5l-15.512-2.093 3.377-4.96-35.428-24.12-.826-.563 1.125-1.653z"/><path id="Rectangle-5" stroke="#A7333A" stroke-width="2" d="M48 112h364v150H48z"/><text id="family:-null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="183" y="66">family: null</tspan></text><g id="noun_48910_cc" transform="translate(8 110)"><path id="Shape" d="M17.503 1.75h-5.006a.341.341 0 00-.34.342v1.125h5.686V2.092a.341.341 0 00-.34-.341z"/><path id="Shape" fill="#A7333A" d="M28.364 3.217H19.59V2.092A2.09 2.09 0 0017.503 0h-5.006c-1.15 0-2.087.938-2.087 2.092v1.125H1.637c-.7 0-1.266.568-1.266 1.269v.09c0 .7.567 1.267 1.266 1.267h26.727c.699 0 1.266-.567 1.266-1.268v-.09c0-.7-.567-1.268-1.266-1.268zm-10.52 0h-5.687V2.092c0-.188.153-.341.34-.341h5.006a.34.34 0 01.34.34v1.126zM26.054 6.281H3.728c-1.298 0-2.35-.224-2.35 1.077L3.14 33.196c0 1.3 1.052 2.409 2.35 2.409h18.802c1.298 0 2.35-1.11 2.35-2.409l1.763-25.838c0-1.301-1.053-1.077-2.35-1.077zM9.637 32.193c-.377.012-.691-.261-.704-.612l-.694-19.917c-.012-.351.283-.647.66-.66.376-.013.69.261.703.613l.694 19.916c.013.351-.283.647-.659.66zm6.044-.63c0 .352-.306.637-.682.637-.377 0-.682-.286-.682-.637V11.634c0-.351.305-.636.682-.636.376 0 .682.285.682.636v19.93zm5.384.018c-.012.351-.327.625-.704.612-.376-.013-.672-.308-.66-.66l.695-19.916c.012-.352.326-.626.703-.613.377.014.672.309.66.66l-.694 19.917z"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/family-no-family.svg diff --git a/1-js/04-object-basics/02-garbage-collection/family-no-father-2.svg b/1-js/04-object-basics/03-garbage-collection/family-no-father-2.svg similarity index 52% rename from 1-js/04-object-basics/02-garbage-collection/family-no-father-2.svg rename to 1-js/04-object-basics/03-garbage-collection/family-no-father-2.svg index 502785902..742509d30 100644 --- a/1-js/04-object-basics/02-garbage-collection/family-no-father-2.svg +++ b/1-js/04-object-basics/03-garbage-collection/family-no-father-2.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/family-no-father-2.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="144px" height="225px" viewBox="0 0 144 225" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -30,4 +31,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="144" height="225" viewBox="0 0 144 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="family-no-father-2.svg"><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M34 91h78v26H34z"/><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="50" y="107">Object</tspan></text><text id="family" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="15" y="67">family</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M16 166h118v48H16z"/><text id="name:-"Ann"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="34" y="203">name: "Ann"</tspan></text><text id="mother" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="16" y="142">mother</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="48" y="183">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M75.5 47.5v18h6l-7 14-7-14h6v-18h2zM74.5 122.5v19h6l-7 14-7-14h6v-19h2z"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M27 16h93v26H27z"/><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="43" y="33"><global></tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/family-no-father-2.svg diff --git a/1-js/04-object-basics/02-garbage-collection/family-no-father.svg b/1-js/04-object-basics/03-garbage-collection/family-no-father.svg similarity index 61% rename from 1-js/04-object-basics/02-garbage-collection/family-no-father.svg rename to 1-js/04-object-basics/03-garbage-collection/family-no-father.svg index 9837ced82..ada595880 100644 --- a/1-js/04-object-basics/02-garbage-collection/family-no-father.svg +++ b/1-js/04-object-basics/03-garbage-collection/family-no-father.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/family-no-father.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="399px" height="225px" viewBox="0 0 399 225" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -47,4 +48,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="399" height="225" viewBox="0 0 399 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="family-no-father.svg"><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M187 78h78v26h-78z"/><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="203" y="94">Object</tspan></text><path id="Rectangle-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M69 146h118v48H69z"/><text id="Object-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="103" y="163">Object</tspan></text><text id="wife" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="211" y="146">wife</tspan></text><text id="family" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="171" y="61">family</tspan></text><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="83" y="183">name: "John"</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M278 146h118v48H278z"/><text id="name:-"Ann"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="299" y="183">name: "Ann"</tspan></text><text id="mother" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="306" y="125">mother</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="308" y="163">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M228.5 41.5v18h6l-7 14-7-14h6v-18h2zM244.5 147.5l14 7-14 7v-6h-48v-2h48v-6zM272.236 109.11l.827.563 35.427 24.121 3.377-4.96L319.5 142.5l-15.512-2.093 3.377-4.96-35.428-24.12-.826-.563 1.125-1.653z"/><path id="Rectangle-5" stroke="#A7333A" stroke-width="2" d="M48 117h217v99H48z"/><g id="noun_48910_cc" transform="translate(7 114)"><path id="Shape" d="M17.503 1.75h-5.006a.341.341 0 00-.34.342v1.125h5.686V2.092a.341.341 0 00-.34-.341z"/><path id="Shape" fill="#A7333A" d="M28.364 3.217H19.59V2.092A2.09 2.09 0 0017.503 0h-5.006c-1.15 0-2.087.938-2.087 2.092v1.125H1.637c-.7 0-1.266.568-1.266 1.269v.09c0 .7.567 1.267 1.266 1.267h26.727c.699 0 1.266-.567 1.266-1.268v-.09c0-.7-.567-1.268-1.266-1.268zm-10.52 0h-5.687V2.092c0-.188.153-.341.34-.341h5.006a.34.34 0 01.34.34v1.126zM26.054 6.281H3.728c-1.298 0-2.35-.224-2.35 1.077L3.14 33.196c0 1.3 1.052 2.409 2.35 2.409h18.802c1.298 0 2.35-1.11 2.35-2.409l1.763-25.838c0-1.301-1.053-1.077-2.35-1.077zM9.637 32.193c-.377.012-.691-.261-.704-.612l-.694-19.917c-.012-.351.283-.647.66-.66.376-.013.69.261.703.613l.694 19.916c.013.351-.283.647-.659.66zm6.044-.63c0 .352-.306.637-.682.637-.377 0-.682-.286-.682-.637V11.634c0-.351.305-.636.682-.636.376 0 .682.285.682.636v19.93zm5.384.018c-.012.351-.327.625-.704.612-.376-.013-.672-.308-.66-.66l.695-19.916c.012-.352.326-.626.703-.613.377.014.672.309.66.66l-.694 19.917z"/></g><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 13h93v26h-93z"/><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="197" y="30"><global></tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/family-no-father.svg diff --git a/1-js/04-object-basics/02-garbage-collection/family.svg b/1-js/04-object-basics/03-garbage-collection/family.svg similarity index 54% rename from 1-js/04-object-basics/02-garbage-collection/family.svg rename to 1-js/04-object-basics/03-garbage-collection/family.svg index 6c66e8569..8ac961be5 100644 --- a/1-js/04-object-basics/02-garbage-collection/family.svg +++ b/1-js/04-object-basics/03-garbage-collection/family.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/family.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="337px" height="204px" viewBox="0 0 337 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -49,4 +50,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="337" height="204" viewBox="0 0 337 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="family.svg"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M147.5 183.5v6h46v2h-46v6l-14-7 14-7z"/><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M123 80h78v26h-78z"/><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="139" y="96">Object</tspan></text><path id="Rectangle-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M5 148h118v48H5z"/><text id="Object-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="39" y="165">Object</tspan></text><text id="father" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="35" y="127">father</tspan></text><text id="wife" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="147" y="148">wife</tspan></text><text id="family" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="107" y="63">family</tspan></text><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="19" y="185">name: "John"</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M214 148h118v48H214z"/><text id="name:-"Ann"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="230" y="185">name: "Ann"</tspan></text><text id="mother" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="242" y="127">mother</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="244" y="165">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M164.5 43.5v18h6l-7 14-7-14h6v-18h2z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M114.75 111.108l1.142 1.642-.82.57-34.508 24.005 3.426 4.926L68.5 144.5l7.495-13.741 3.426 4.924 34.508-24.004.82-.571z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M180.5 149.5l14 7-14 7v-6h-48v-2h48v-6z"/><text id="husband" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="136" y="181">husband</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M208.236 111.11l.827.563 35.427 24.121 3.377-4.96L255.5 144.5l-15.512-2.093 3.377-4.96-35.428-24.12-.826-.563 1.125-1.653z"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M88 13h151v26H88z"/><text id="<global-variable>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="92" y="30"><global variable></tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/family.svg diff --git a/1-js/04-object-basics/02-garbage-collection/garbage-collection-1.svg b/1-js/04-object-basics/03-garbage-collection/garbage-collection-1.svg similarity index 59% rename from 1-js/04-object-basics/02-garbage-collection/garbage-collection-1.svg rename to 1-js/04-object-basics/03-garbage-collection/garbage-collection-1.svg index d3bc5ce94..68e20df3b 100644 --- a/1-js/04-object-basics/02-garbage-collection/garbage-collection-1.svg +++ b/1-js/04-object-basics/03-garbage-collection/garbage-collection-1.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/garbage-collection-1.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="204px" viewBox="0 0 463 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -159,4 +160,7 @@ <path id="Line-Copy" d="M367.026599,89.8074683 L370.905564,94.6561738 L369.343826,95.9055639 L365.464862,91.0568584 L360.779649,94.8050287 L357.5,79.5 L371.711812,86.059298 L367.026599,89.8074683 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="463" height="204" viewBox="0 0 463 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M0 0h80v28H0z"/><path id="path-3" d="M0 0h40v20H0z"/><path id="path-5" d="M0 0h40v20H0z"/><path id="path-7" d="M0 0h40v20H0z"/><path id="path-9" d="M0 0h40v20H0z"/><path id="path-11" d="M0 0h40v20H0z"/><path id="path-13" d="M0 0h40v20H0z"/><path id="path-15" d="M0 0h40v20H0z"/><path id="path-17" d="M0 0h40v20H0z"/><path id="path-19" d="M0 0h40v20H0z"/><path id="path-21" d="M0 0h40v20H0z"/><path id="path-23" d="M0 0h40v20H0z"/></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="garbage-collection-1.svg"><g id="Rectangle-1-Clipped" transform="translate(106 8)"><mask id="mask-2" fill="#fff"><use xlink:href="#path-1"/></mask><g id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-2)"><path id="path-1" d="M0 0h80v28H0z"/></g></g><text id="<global>" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="26"><global></tspan></text><g id="Rectangle-2-Clipped" transform="translate(360 99)"><mask id="mask-4" fill="#fff"><use xlink:href="#path-3"/></mask><g id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-4)"><path id="path-3" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(267 99)"><mask id="mask-6" fill="#fff"><use xlink:href="#path-5"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-6)"><path id="path-5" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(126 79)"><mask id="mask-8" fill="#fff"><use xlink:href="#path-7"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-8)"><path id="path-7" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-Clipped" transform="translate(313 53)"><mask id="mask-10" fill="#fff"><use xlink:href="#path-9"/></mask><g id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-10)"><path id="path-9" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-3-Clipped" transform="translate(176 129)"><mask id="mask-12" fill="#fff"><use xlink:href="#path-11"/></mask><g id="Rectangle-2-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-12)"><path id="path-11" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-8-Clipped" transform="translate(196 69)"><mask id="mask-14" fill="#fff"><use xlink:href="#path-13"/></mask><g id="Rectangle-2-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-14)"><path id="path-13" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-4-Clipped" transform="translate(128 139)"><mask id="mask-16" fill="#fff"><use xlink:href="#path-15"/></mask><g id="Rectangle-2-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-16)"><path id="path-15" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-5-Clipped" transform="translate(76 131)"><mask id="mask-18" fill="#fff"><use xlink:href="#path-17"/></mask><g id="Rectangle-2-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-18)"><path id="path-17" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-6-Clipped" transform="translate(76 71)"><mask id="mask-20" fill="#fff"><use xlink:href="#path-19"/></mask><g id="Rectangle-2-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-20)"><path id="path-19" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-7-Clipped" transform="translate(27 57)"><mask id="mask-22" fill="#fff"><use xlink:href="#path-21"/></mask><g id="Rectangle-2-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-22)"><path id="path-21" d="M0 0h40v20H0z"/></g></g><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 40.5v31"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 71.5l3-10.8h-6z"/><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 105.5v24"/><path id="Line-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 129.5l3-10.8h-6z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M319.5 77.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M300.5 93.5l10.193-4.662-3.864-4.59z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M127.5 47.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M108.5 63.5l10.193-4.662-3.864-4.59z"/><path id="Line-2-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M97.5 42.5l-23 9"/><path id="Line-2-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M74.5 51.5l11.15-1.142-2.186-5.587z"/><path id="Line-2-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M133.5 107.5l-19 16"/><path id="Line-2-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M114.5 123.5l10.193-4.662-3.864-4.59z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M313.071 109.5h39.354"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M352.425 109.5l-10.8-3v6z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 107.5l18 15"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M188.5 122.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-width="2" d="M202.5 154.5l18 15"/><path id="Line-Copy-6-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M220.5 169.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-5" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 87.5l20-7"/><path id="Line-Copy-5-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M190.5 80.5l-11.185.736 1.982 5.663z"/><path id="Line-Copy-4" fill="#C06334" stroke="#C06334" stroke-width="2" d="M209.5 120.5l14-22"/><path id="Line-Copy-4-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M223.5 98.5l-8.33 7.5 5.063 3.222z"/><path id="Line-Copy-3" fill="#C06334" stroke="#C06334" stroke-width="2" d="M46.5 82.5l32 40"/><path id="Line-Copy-3-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M78.5 122.5l-4.404-10.307-4.685 3.748z"/><g id="Rectangle-2-Copy-9-Clipped" transform="translate(219 177)"><mask id="mask-24" fill="#fff"><use xlink:href="#path-23"/></mask><g id="Rectangle-2-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-24)"><path id="path-25" d="M0 0h40v20H0z"/></g></g><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M187.5 41.5l21 21"/><path id="Line-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M208.5 62.5l-5.515-9.758-4.243 4.243z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M357.5 79.5l14.212 6.56-4.686 3.747 3.255 4.068.625.781-1.562 1.25-.625-.781-3.255-4.069-4.684 3.749L357.5 79.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/garbage-collection-1.svg diff --git a/1-js/04-object-basics/02-garbage-collection/garbage-collection-2.svg b/1-js/04-object-basics/03-garbage-collection/garbage-collection-2.svg similarity index 62% rename from 1-js/04-object-basics/02-garbage-collection/garbage-collection-2.svg rename to 1-js/04-object-basics/03-garbage-collection/garbage-collection-2.svg index c7311022a..aaac31b9e 100644 --- a/1-js/04-object-basics/02-garbage-collection/garbage-collection-2.svg +++ b/1-js/04-object-basics/03-garbage-collection/garbage-collection-2.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/garbage-collection-2.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="204px" viewBox="0 0 463 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -175,4 +176,7 @@ <path id="Line-Copy" d="M367.026599,89.8074683 L370.905564,94.6561738 L369.343826,95.9055639 L365.464862,91.0568584 L360.779649,94.8050287 L357.5,79.5 L371.711812,86.059298 L367.026599,89.8074683 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="463" height="204" viewBox="0 0 463 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M0 0h80v28H0z"/><path id="path-3" d="M0 0h40v20H0z"/><path id="path-5" d="M0 0h40v20H0z"/><path id="path-7" d="M0 0h40v20H0z"/><path id="path-9" d="M0 0h40v20H0z"/><path id="path-11" d="M0 0h40v20H0z"/><path id="path-13" d="M0 0h40v20H0z"/><path id="path-15" d="M0 0h40v20H0z"/><path id="path-17" d="M0 0h40v20H0z"/><path id="path-19" d="M0 0h40v20H0z"/><path id="path-21" d="M0 0h40v20H0z"/><path id="path-23" d="M0 0h40v20H0z"/></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="garbage-collection-2.svg"><g id="Rectangle-1-Clipped" transform="translate(106 8)"><mask id="mask-2" fill="#fff"><use xlink:href="#path-1"/></mask><g id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-2)"><path id="path-1" d="M0 0h80v28H0z"/></g></g><text id="<global>" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="26"><global></tspan></text><g id="Rectangle-2-Clipped" transform="translate(360 99)"><mask id="mask-4" fill="#fff"><use xlink:href="#path-3"/></mask><g id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-4)"><path id="path-3" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(267 99)"><mask id="mask-6" fill="#fff"><use xlink:href="#path-5"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-6)"><path id="path-5" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(126 79)"><mask id="mask-8" fill="#fff"><use xlink:href="#path-7"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-8)"><path id="path-7" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-Clipped" transform="translate(313 53)"><mask id="mask-10" fill="#fff"><use xlink:href="#path-9"/></mask><g id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-10)"><path id="path-9" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-3-Clipped" transform="translate(176 129)"><mask id="mask-12" fill="#fff"><use xlink:href="#path-11"/></mask><g id="Rectangle-2-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-12)"><path id="path-11" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-8-Clipped" transform="translate(196 69)"><mask id="mask-14" fill="#fff"><use xlink:href="#path-13"/></mask><g id="Rectangle-2-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-14)"><path id="path-13" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-4-Clipped" transform="translate(128 139)"><mask id="mask-16" fill="#fff"><use xlink:href="#path-15"/></mask><g id="Rectangle-2-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-16)"><path id="path-15" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-5-Clipped" transform="translate(76 131)"><mask id="mask-18" fill="#fff"><use xlink:href="#path-17"/></mask><g id="Rectangle-2-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-18)"><path id="path-17" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-6-Clipped" transform="translate(76 71)"><mask id="mask-20" fill="#fff"><use xlink:href="#path-19"/></mask><g id="Rectangle-2-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-20)"><path id="path-19" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-7-Clipped" transform="translate(27 57)"><mask id="mask-22" fill="#fff"><use xlink:href="#path-21"/></mask><g id="Rectangle-2-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-22)"><path id="path-21" d="M0 0h40v20H0z"/></g></g><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 40.5v31"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 71.5l3-10.8h-6z"/><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 105.5v24"/><path id="Line-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 129.5l3-10.8h-6z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M319.5 77.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M300.5 93.5l10.193-4.662-3.864-4.59z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M127.5 47.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M108.5 63.5l10.193-4.662-3.864-4.59z"/><path id="Line-2-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M97.5 42.5l-23 9"/><path id="Line-2-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M74.5 51.5l11.15-1.142-2.186-5.587z"/><path id="Line-2-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M133.5 107.5l-19 16"/><path id="Line-2-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M114.5 123.5l10.193-4.662-3.864-4.59z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M313.071 109.5h39.354"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M352.425 109.5l-10.8-3v6z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 107.5l18 15"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M188.5 122.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-width="2" d="M202.5 154.5l18 15"/><path id="Line-Copy-6-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M220.5 169.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-5" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 87.5l20-7"/><path id="Line-Copy-5-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M190.5 80.5l-11.185.736 1.982 5.663z"/><path id="Line-Copy-4" fill="#C06334" stroke="#C06334" stroke-width="2" d="M209.5 120.5l14-22"/><path id="Line-Copy-4-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M223.5 98.5l-8.33 7.5 5.063 3.222z"/><path id="Line-Copy-3" fill="#C06334" stroke="#C06334" stroke-width="2" d="M46.5 82.5l32 40"/><path id="Line-Copy-3-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M78.5 122.5l-4.404-10.307-4.685 3.748z"/><g id="Group" fill="#478964" transform="rotate(-90 81 2)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy" fill="#478964" transform="rotate(-90 49.5 19.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-4" fill="#478964" transform="rotate(-90 110 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Rectangle-2-Copy-9-Clipped" transform="translate(219 177)"><mask id="mask-24" fill="#fff"><use xlink:href="#path-23"/></mask><g id="Rectangle-2-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-24)"><path id="path-25" d="M0 0h40v20H0z"/></g></g><g id="Group-Copy-6" fill="#478964" transform="rotate(-90 140 -59)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M187.5 41.5l21 21"/><path id="Line-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M208.5 62.5l-5.515-9.758-4.243 4.243z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M357.5 79.5l14.212 6.56-4.686 3.747 3.255 4.068.625.781-1.562 1.25-.625-.781-3.255-4.069-4.684 3.749L357.5 79.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/garbage-collection-2.svg diff --git a/1-js/04-object-basics/02-garbage-collection/garbage-collection-3.svg b/1-js/04-object-basics/03-garbage-collection/garbage-collection-3.svg similarity index 63% rename from 1-js/04-object-basics/02-garbage-collection/garbage-collection-3.svg rename to 1-js/04-object-basics/03-garbage-collection/garbage-collection-3.svg index a0ce257e3..523043c83 100644 --- a/1-js/04-object-basics/02-garbage-collection/garbage-collection-3.svg +++ b/1-js/04-object-basics/03-garbage-collection/garbage-collection-3.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/garbage-collection-3.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="204px" viewBox="0 0 463 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -187,4 +188,7 @@ <path id="Line-Copy" d="M367.026599,89.8074683 L370.905564,94.6561738 L369.343826,95.9055639 L365.464862,91.0568584 L360.779649,94.8050287 L357.5,79.5 L371.711812,86.059298 L367.026599,89.8074683 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="463" height="204" viewBox="0 0 463 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M0 0h80v28H0z"/><path id="path-3" d="M0 0h40v20H0z"/><path id="path-5" d="M0 0h40v20H0z"/><path id="path-7" d="M0 0h40v20H0z"/><path id="path-9" d="M0 0h40v20H0z"/><path id="path-11" d="M0 0h40v20H0z"/><path id="path-13" d="M0 0h40v20H0z"/><path id="path-15" d="M0 0h40v20H0z"/><path id="path-17" d="M0 0h40v20H0z"/><path id="path-19" d="M0 0h40v20H0z"/><path id="path-21" d="M0 0h40v20H0z"/><path id="path-23" d="M0 0h40v20H0z"/></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="garbage-collection-3.svg"><g id="Rectangle-1-Clipped" transform="translate(106 8)"><mask id="mask-2" fill="#fff"><use xlink:href="#path-1"/></mask><g id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-2)"><path id="path-1" d="M0 0h80v28H0z"/></g></g><text id="<global>" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="26"><global></tspan></text><g id="Rectangle-2-Clipped" transform="translate(360 99)"><mask id="mask-4" fill="#fff"><use xlink:href="#path-3"/></mask><g id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-4)"><path id="path-3" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(267 99)"><mask id="mask-6" fill="#fff"><use xlink:href="#path-5"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-6)"><path id="path-5" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(126 79)"><mask id="mask-8" fill="#fff"><use xlink:href="#path-7"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-8)"><path id="path-7" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-Clipped" transform="translate(313 53)"><mask id="mask-10" fill="#fff"><use xlink:href="#path-9"/></mask><g id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-10)"><path id="path-9" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-3-Clipped" transform="translate(176 129)"><mask id="mask-12" fill="#fff"><use xlink:href="#path-11"/></mask><g id="Rectangle-2-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-12)"><path id="path-11" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-8-Clipped" transform="translate(196 69)"><mask id="mask-14" fill="#fff"><use xlink:href="#path-13"/></mask><g id="Rectangle-2-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-14)"><path id="path-13" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-4-Clipped" transform="translate(128 139)"><mask id="mask-16" fill="#fff"><use xlink:href="#path-15"/></mask><g id="Rectangle-2-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-16)"><path id="path-15" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-5-Clipped" transform="translate(76 131)"><mask id="mask-18" fill="#fff"><use xlink:href="#path-17"/></mask><g id="Rectangle-2-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-18)"><path id="path-17" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-6-Clipped" transform="translate(76 71)"><mask id="mask-20" fill="#fff"><use xlink:href="#path-19"/></mask><g id="Rectangle-2-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-20)"><path id="path-19" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-7-Clipped" transform="translate(27 57)"><mask id="mask-22" fill="#fff"><use xlink:href="#path-21"/></mask><g id="Rectangle-2-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-22)"><path id="path-21" d="M0 0h40v20H0z"/></g></g><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 40.5v31"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 71.5l3-10.8h-6z"/><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 105.5v24"/><path id="Line-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 129.5l3-10.8h-6z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M319.5 77.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M300.5 93.5l10.193-4.662-3.864-4.59z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M127.5 47.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M108.5 63.5l10.193-4.662-3.864-4.59z"/><path id="Line-2-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M97.5 42.5l-23 9"/><path id="Line-2-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M74.5 51.5l11.15-1.142-2.186-5.587z"/><path id="Line-2-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M133.5 107.5l-19 16"/><path id="Line-2-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M114.5 123.5l10.193-4.662-3.864-4.59z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M313.071 109.5h39.354"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M352.425 109.5l-10.8-3v6z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 107.5l18 15"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M188.5 122.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-width="2" d="M202.5 154.5l18 15"/><path id="Line-Copy-6-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M220.5 169.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-5" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 87.5l20-7"/><path id="Line-Copy-5-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M190.5 80.5l-11.185.736 1.982 5.663z"/><path id="Line-Copy-4" fill="#C06334" stroke="#C06334" stroke-width="2" d="M209.5 120.5l14-22"/><path id="Line-Copy-4-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M223.5 98.5l-8.33 7.5 5.063 3.222z"/><path id="Line-Copy-3" fill="#C06334" stroke="#C06334" stroke-width="2" d="M46.5 82.5l32 40"/><path id="Line-Copy-3-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M78.5 122.5l-4.404-10.307-4.685 3.748z"/><g id="Group" fill="#478964" transform="rotate(-90 81 2)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy" fill="#478964" transform="rotate(-90 49.5 19.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-2" fill="#478964" transform="rotate(-90 111.5 32.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-3" fill="#478964" transform="rotate(-90 141 10)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-4" fill="#478964" transform="rotate(-90 110 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-5" fill="#478964" transform="rotate(-90 160 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Rectangle-2-Copy-9-Clipped" transform="translate(219 177)"><mask id="mask-24" fill="#fff"><use xlink:href="#path-23"/></mask><g id="Rectangle-2-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-24)"><path id="path-25" d="M0 0h40v20H0z"/></g></g><g id="Group-Copy-6" fill="#478964" transform="rotate(-90 140 -59)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M187.5 41.5l21 21"/><path id="Line-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M208.5 62.5l-5.515-9.758-4.243 4.243z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M357.5 79.5l14.212 6.56-4.686 3.747 3.255 4.068.625.781-1.562 1.25-.625-.781-3.255-4.069-4.684 3.749L357.5 79.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/garbage-collection-3.svg diff --git a/1-js/04-object-basics/02-garbage-collection/garbage-collection-4.svg b/1-js/04-object-basics/03-garbage-collection/garbage-collection-4.svg similarity index 63% rename from 1-js/04-object-basics/02-garbage-collection/garbage-collection-4.svg rename to 1-js/04-object-basics/03-garbage-collection/garbage-collection-4.svg index bc5a035c7..201576082 100644 --- a/1-js/04-object-basics/02-garbage-collection/garbage-collection-4.svg +++ b/1-js/04-object-basics/03-garbage-collection/garbage-collection-4.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/garbage-collection-4.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="204px" viewBox="0 0 463 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -191,4 +192,7 @@ <path id="Line-Copy" d="M367.026599,89.8074683 L370.905564,94.6561738 L369.343826,95.9055639 L365.464862,91.0568584 L360.779649,94.8050287 L357.5,79.5 L371.711812,86.059298 L367.026599,89.8074683 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="463" height="204" viewBox="0 0 463 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M0 0h80v28H0z"/><path id="path-3" d="M0 0h40v20H0z"/><path id="path-5" d="M0 0h40v20H0z"/><path id="path-7" d="M0 0h40v20H0z"/><path id="path-9" d="M0 0h40v20H0z"/><path id="path-11" d="M0 0h40v20H0z"/><path id="path-13" d="M0 0h40v20H0z"/><path id="path-15" d="M0 0h40v20H0z"/><path id="path-17" d="M0 0h40v20H0z"/><path id="path-19" d="M0 0h40v20H0z"/><path id="path-21" d="M0 0h40v20H0z"/><path id="path-23" d="M0 0h40v20H0z"/></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="garbage-collection-4.svg"><g id="Rectangle-1-Clipped" transform="translate(106 8)"><mask id="mask-2" fill="#fff"><use xlink:href="#path-1"/></mask><g id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-2)"><path id="path-1" d="M0 0h80v28H0z"/></g></g><text id="<global>" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="26"><global></tspan></text><g id="Rectangle-2-Clipped" transform="translate(360 99)"><mask id="mask-4" fill="#fff"><use xlink:href="#path-3"/></mask><g id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-4)"><path id="path-3" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(267 99)"><mask id="mask-6" fill="#fff"><use xlink:href="#path-5"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-6)"><path id="path-5" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(126 79)"><mask id="mask-8" fill="#fff"><use xlink:href="#path-7"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-8)"><path id="path-7" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-Clipped" transform="translate(313 53)"><mask id="mask-10" fill="#fff"><use xlink:href="#path-9"/></mask><g id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-10)"><path id="path-9" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-3-Clipped" transform="translate(176 129)"><mask id="mask-12" fill="#fff"><use xlink:href="#path-11"/></mask><g id="Rectangle-2-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-12)"><path id="path-11" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-8-Clipped" transform="translate(196 69)"><mask id="mask-14" fill="#fff"><use xlink:href="#path-13"/></mask><g id="Rectangle-2-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-14)"><path id="path-13" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-4-Clipped" transform="translate(128 139)"><mask id="mask-16" fill="#fff"><use xlink:href="#path-15"/></mask><g id="Rectangle-2-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-16)"><path id="path-15" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-5-Clipped" transform="translate(76 131)"><mask id="mask-18" fill="#fff"><use xlink:href="#path-17"/></mask><g id="Rectangle-2-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-18)"><path id="path-17" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-6-Clipped" transform="translate(76 71)"><mask id="mask-20" fill="#fff"><use xlink:href="#path-19"/></mask><g id="Rectangle-2-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-20)"><path id="path-19" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-7-Clipped" transform="translate(27 57)"><mask id="mask-22" fill="#fff"><use xlink:href="#path-21"/></mask><g id="Rectangle-2-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-22)"><path id="path-21" d="M0 0h40v20H0z"/></g></g><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 40.5v31"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 71.5l3-10.8h-6z"/><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 105.5v24"/><path id="Line-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 129.5l3-10.8h-6z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M319.5 77.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M300.5 93.5l10.193-4.662-3.864-4.59z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M127.5 47.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M108.5 63.5l10.193-4.662-3.864-4.59z"/><path id="Line-2-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M97.5 42.5l-23 9"/><path id="Line-2-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M74.5 51.5l11.15-1.142-2.186-5.587z"/><path id="Line-2-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M133.5 107.5l-19 16"/><path id="Line-2-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M114.5 123.5l10.193-4.662-3.864-4.59z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M313.071 109.5h39.354"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M352.425 109.5l-10.8-3v6z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 107.5l18 15"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M188.5 122.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-width="2" d="M202.5 154.5l18 15"/><path id="Line-Copy-6-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M220.5 169.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-5" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 87.5l20-7"/><path id="Line-Copy-5-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M190.5 80.5l-11.185.736 1.982 5.663z"/><path id="Line-Copy-4" fill="#C06334" stroke="#C06334" stroke-width="2" d="M209.5 120.5l14-22"/><path id="Line-Copy-4-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M223.5 98.5l-8.33 7.5 5.063 3.222z"/><path id="Line-Copy-3" fill="#C06334" stroke="#C06334" stroke-width="2" d="M46.5 82.5l32 40"/><path id="Line-Copy-3-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M78.5 122.5l-4.404-10.307-4.685 3.748z"/><g id="Group" fill="#478964" transform="rotate(-90 81 2)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy" fill="#478964" transform="rotate(-90 49.5 19.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-2" fill="#478964" transform="rotate(-90 111.5 32.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-3" fill="#478964" transform="rotate(-90 141 10)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-4" fill="#478964" transform="rotate(-90 110 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-5" fill="#478964" transform="rotate(-90 160 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Rectangle-2-Copy-9-Clipped" transform="translate(219 177)"><mask id="mask-24" fill="#fff"><use xlink:href="#path-23"/></mask><g id="Rectangle-2-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-24)"><path id="path-25" d="M0 0h40v20H0z"/></g></g><g id="Group-Copy-7" fill="#478964" transform="rotate(-90 205.5 -16.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-6" fill="#478964" transform="rotate(-90 140 -59)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M187.5 41.5l21 21"/><path id="Line-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M208.5 62.5l-5.515-9.758-4.243 4.243z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M357.5 79.5l14.212 6.56-4.686 3.747 3.255 4.068.625.781-1.562 1.25-.625-.781-3.255-4.069-4.684 3.749L357.5 79.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/garbage-collection-4.svg diff --git a/1-js/04-object-basics/02-garbage-collection/garbage-collection-5.svg b/1-js/04-object-basics/03-garbage-collection/garbage-collection-5.svg similarity index 64% rename from 1-js/04-object-basics/02-garbage-collection/garbage-collection-5.svg rename to 1-js/04-object-basics/03-garbage-collection/garbage-collection-5.svg index 150aa9265..906235438 100644 --- a/1-js/04-object-basics/02-garbage-collection/garbage-collection-5.svg +++ b/1-js/04-object-basics/03-garbage-collection/garbage-collection-5.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/garbage-collection-5.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="204px" viewBox="0 0 463 204" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -209,4 +210,7 @@ <path id="Line" d="M367.026599,89.8074683 L370.905564,94.6561738 L369.343826,95.9055639 L365.464862,91.0568584 L360.779649,94.8050287 L357.5,79.5 L371.711812,86.059298 L367.026599,89.8074683 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="463" height="204" viewBox="0 0 463 204"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M0 0h146v88H0z"/><path id="path-3" d="M0 0h80v28H0z"/><path id="path-5" d="M0 0h40v20H0z"/><path id="path-7" d="M0 0h40v20H0z"/><path id="path-9" d="M0 0h40v20H0z"/><path id="path-11" d="M0 0h40v20H0z"/><path id="path-13" d="M0 0h40v20H0z"/><path id="path-15" d="M0 0h40v20H0z"/><path id="path-17" d="M0 0h40v20H0z"/><path id="path-19" d="M0 0h40v20H0z"/><path id="path-21" d="M0 0h40v20H0z"/><path id="path-23" d="M0 0h40v20H0z"/><path id="path-25" d="M0 0h40v20H0z"/></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="garbage-collection-5.svg"><g id="Rectangle-5-Clipped" transform="translate(260 46)"><mask id="mask-2" fill="#fff"><use xlink:href="#path-1"/></mask><g id="Rectangle-5" stroke="#A7333A" stroke-width="4" mask="url(#mask-2)"><path id="path-23" d="M0 0h146v88H0z"/></g></g><g id="Rectangle-1-Clipped" transform="translate(106 8)"><mask id="mask-4" fill="#fff"><use xlink:href="#path-3"/></mask><g id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-4)"><path id="path-1" d="M0 0h80v28H0z"/></g></g><text id="<global>" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="26"><global></tspan></text><g id="Rectangle-2-Clipped" transform="translate(360 99)"><mask id="mask-6" fill="#fff"><use xlink:href="#path-5"/></mask><g id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-6)"><path id="path-3" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(267 99)"><mask id="mask-8" fill="#fff"><use xlink:href="#path-7"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-8)"><path id="path-5" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-2-Clipped" transform="translate(126 79)"><mask id="mask-10" fill="#fff"><use xlink:href="#path-9"/></mask><g id="Rectangle-2-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-10)"><path id="path-7" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-Clipped" transform="translate(313 53)"><mask id="mask-12" fill="#fff"><use xlink:href="#path-11"/></mask><g id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-12)"><path id="path-9" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-3-Clipped" transform="translate(176 129)"><mask id="mask-14" fill="#fff"><use xlink:href="#path-13"/></mask><g id="Rectangle-2-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-14)"><path id="path-11" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-8-Clipped" transform="translate(196 69)"><mask id="mask-16" fill="#fff"><use xlink:href="#path-15"/></mask><g id="Rectangle-2-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-16)"><path id="path-13" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-4-Clipped" transform="translate(128 139)"><mask id="mask-18" fill="#fff"><use xlink:href="#path-17"/></mask><g id="Rectangle-2-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-18)"><path id="path-15" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-5-Clipped" transform="translate(76 131)"><mask id="mask-20" fill="#fff"><use xlink:href="#path-19"/></mask><g id="Rectangle-2-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-20)"><path id="path-17" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-6-Clipped" transform="translate(76 71)"><mask id="mask-22" fill="#fff"><use xlink:href="#path-21"/></mask><g id="Rectangle-2-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-22)"><path id="path-19" d="M0 0h40v20H0z"/></g></g><g id="Rectangle-2-Copy-7-Clipped" transform="translate(27 57)"><mask id="mask-24" fill="#fff"><use xlink:href="#path-23"/></mask><g id="Rectangle-2-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-24)"><path id="path-21" d="M0 0h40v20H0z"/></g></g><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 40.5v31"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 71.5l3-10.8h-6z"/><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 105.5v24"/><path id="Line-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M148.5 129.5l3-10.8h-6z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M319.5 77.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M300.5 93.5l10.193-4.662-3.864-4.59z"/><path id="Line-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M127.5 47.5l-19 16"/><path id="Line-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M108.5 63.5l10.193-4.662-3.864-4.59z"/><path id="Line-2-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M97.5 42.5l-23 9"/><path id="Line-2-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M74.5 51.5l11.15-1.142-2.186-5.587z"/><path id="Line-2-Copy" fill="#C06334" stroke="#C06334" stroke-width="2" d="M133.5 107.5l-19 16"/><path id="Line-2-Copy-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M114.5 123.5l10.193-4.662-3.864-4.59z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M313.071 109.5h39.354"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M352.425 109.5l-10.8-3v6z"/><path id="Line" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 107.5l18 15"/><path id="Line-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M188.5 122.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-width="2" d="M202.5 154.5l18 15"/><path id="Line-Copy-6-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M220.5 169.5l-6.376-9.219-3.841 4.61z"/><path id="Line-Copy-5" fill="#C06334" stroke="#C06334" stroke-width="2" d="M170.5 87.5l20-7"/><path id="Line-Copy-5-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M190.5 80.5l-11.185.736 1.982 5.663z"/><path id="Line-Copy-4" fill="#C06334" stroke="#C06334" stroke-width="2" d="M209.5 120.5l14-22"/><path id="Line-Copy-4-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M223.5 98.5l-8.33 7.5 5.063 3.222z"/><path id="Line-Copy-3" fill="#C06334" stroke="#C06334" stroke-width="2" d="M46.5 82.5l32 40"/><path id="Line-Copy-3-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M78.5 122.5l-4.404-10.307-4.685 3.748z"/><g id="noun_48910_cc" transform="translate(261 5)"><path id="Shape" d="M17.503 1.75h-5.006a.341.341 0 00-.34.342v1.125h5.686V2.092a.341.341 0 00-.34-.341z"/><path id="Shape" fill="#A7333A" d="M28.364 3.217H19.59V2.092A2.09 2.09 0 0017.503 0h-5.006c-1.15 0-2.087.938-2.087 2.092v1.125H1.637c-.7 0-1.266.568-1.266 1.269v.09c0 .7.567 1.267 1.266 1.267h26.727c.699 0 1.266-.567 1.266-1.268v-.09c0-.7-.567-1.268-1.266-1.268zm-10.52 0h-5.687V2.092c0-.188.153-.341.34-.341h5.006a.34.34 0 01.34.34v1.126zM26.054 6.281H3.728c-1.298 0-2.35-.224-2.35 1.077L3.14 33.196c0 1.3 1.052 2.409 2.35 2.409h18.802c1.298 0 2.35-1.11 2.35-2.409l1.763-25.838c0-1.301-1.053-1.077-2.35-1.077zM9.637 32.193c-.377.012-.691-.261-.704-.612l-.694-19.917c-.012-.351.283-.647.66-.66.376-.013.69.261.703.613l.694 19.916c.013.351-.283.647-.659.66zm6.044-.63c0 .352-.306.637-.682.637-.377 0-.682-.286-.682-.637V11.634c0-.351.305-.636.682-.636.376 0 .682.285.682.636v19.93zm5.384.018c-.012.351-.327.625-.704.612-.376-.013-.672-.308-.66-.66l.695-19.916c.012-.352.326-.626.703-.613.377.014.672.309.66.66l-.694 19.917z"/></g><text id="unreachables" fill="#AF6E24" fill-rule="nonzero" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="296" y="29">unreachables</tspan></text><g id="Group" fill="#478964" transform="rotate(-90 81 2)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy" fill="#478964" transform="rotate(-90 49.5 19.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-2" fill="#478964" transform="rotate(-90 111.5 32.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-3" fill="#478964" transform="rotate(-90 141 10)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-4" fill="#478964" transform="rotate(-90 110 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-5" fill="#478964" transform="rotate(-90 160 -19)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Rectangle-2-Copy-9-Clipped" transform="translate(219 177)"><mask id="mask-26" fill="#fff"><use xlink:href="#path-25"/></mask><g id="Rectangle-2-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="4" mask="url(#mask-26)"><path id="path-25" d="M0 0h40v20H0z"/></g></g><g id="Group-Copy-7" fill="#478964" transform="rotate(-90 205.5 -16.5)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><g id="Group-Copy-6" fill="#478964" transform="rotate(-90 140 -59)"><path id="Fill-72" d="M5.5 5.511l-.866-1.01.866-1.012V1L2.505 4.497H2.5l.003.003-.003.003h.005L5.5 8V5.511"/><path id="Fill-73" d="M4.5 1.098A3.406 3.406 0 001.098 4.5 3.406 3.406 0 004.5 7.902 3.406 3.406 0 007.902 4.5 3.406 3.406 0 004.5 1.098zM4.5 9A4.505 4.505 0 010 4.5C0 2.019 2.019 0 4.5 0S9 2.019 9 4.5 6.981 9 4.5 9z"/></g><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-width="2" d="M187.5 41.5l21 21"/><path id="Line-Copy-2-decoration-1" fill="#C06334" stroke="#C06334" stroke-width="2" d="M208.5 62.5l-5.515-9.758-4.243 4.243z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M357.5 79.5l14.212 6.56-4.686 3.747 3.255 4.068.625.781-1.562 1.25-.625-.781-3.255-4.069-4.684 3.749L357.5 79.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/garbage-collection-5.svg diff --git a/1-js/04-object-basics/02-garbage-collection/memory-user-john-admin.svg b/1-js/04-object-basics/03-garbage-collection/memory-user-john-admin.svg similarity index 51% rename from 1-js/04-object-basics/02-garbage-collection/memory-user-john-admin.svg rename to 1-js/04-object-basics/03-garbage-collection/memory-user-john-admin.svg index bc7a23d33..948f55005 100644 --- a/1-js/04-object-basics/02-garbage-collection/memory-user-john-admin.svg +++ b/1-js/04-object-basics/03-garbage-collection/memory-user-john-admin.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/memory-user-john-admin.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="144px" height="159px" viewBox="0 0 144 159" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -26,4 +27,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="144" height="159" viewBox="0 0 144 159"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="memory-user-john-admin.svg"><text id="user" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="67">user</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M16 93h118v48H16z"/><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="30" y="130">name: "John"</tspan></text><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="44" y="110">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M55.5 47.5v18h6l-7 14-7-14h6v-18h2z"/><text id="admin" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="92" y="67">admin</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M83.5 47.5v18h6l-7 14-7-14h6v-18h2z"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M27 16h93v26H27z"/><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="43" y="33"><global></tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/memory-user-john-admin.svg diff --git a/1-js/04-object-basics/02-garbage-collection/memory-user-john-lost.svg b/1-js/04-object-basics/03-garbage-collection/memory-user-john-lost.svg similarity index 61% rename from 1-js/04-object-basics/02-garbage-collection/memory-user-john-lost.svg rename to 1-js/04-object-basics/03-garbage-collection/memory-user-john-lost.svg index 78009e310..d4b301fcc 100644 --- a/1-js/04-object-basics/02-garbage-collection/memory-user-john-lost.svg +++ b/1-js/04-object-basics/03-garbage-collection/memory-user-john-lost.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/memory-user-john-lost.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="225px" height="159px" viewBox="0 0 225 159" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -27,4 +28,7 @@ </g> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="225" height="159" viewBox="0 0 225 159"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="memory-user-john-lost.svg"><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M61 89h118v48H61z"/><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="83" y="126">name: "John"</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="97" y="106">Object</tspan></text><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M65 4h104v42H65z"/><text id="user:-null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="79" y="40">user: null</tspan></text><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="86" y="23"><global></tspan></text><path id="Rectangle-5" stroke="#A7333A" stroke-width="2" d="M47 69h154v86H47z"/><g id="noun_48910_cc" transform="translate(8 65)"><path id="Shape" d="M17.503 1.75h-5.006a.341.341 0 00-.34.342v1.125h5.686V2.092a.341.341 0 00-.34-.341z"/><path id="Shape" fill="#A7333A" d="M28.364 3.217H19.59V2.092A2.09 2.09 0 0017.503 0h-5.006c-1.15 0-2.087.938-2.087 2.092v1.125H1.637c-.7 0-1.266.568-1.266 1.269v.09c0 .7.567 1.267 1.266 1.267h26.727c.699 0 1.266-.567 1.266-1.268v-.09c0-.7-.567-1.268-1.266-1.268zm-10.52 0h-5.687V2.092c0-.188.153-.341.34-.341h5.006a.34.34 0 01.34.34v1.126zM26.054 6.281H3.728c-1.298 0-2.35-.224-2.35 1.077L3.14 33.196c0 1.3 1.052 2.409 2.35 2.409h18.802c1.298 0 2.35-1.11 2.35-2.409l1.763-25.838c0-1.301-1.053-1.077-2.35-1.077zM9.637 32.193c-.377.012-.691-.261-.704-.612l-.694-19.917c-.012-.351.283-.647.66-.66.376-.013.69.261.703.613l.694 19.916c.013.351-.283.647-.659.66zm6.044-.63c0 .352-.306.637-.682.637-.377 0-.682-.286-.682-.637V11.634c0-.351.305-.636.682-.636.376 0 .682.285.682.636v19.93zm5.384.018c-.012.351-.327.625-.704.612-.376-.013-.672-.308-.66-.66l.695-19.916c.012-.352.326-.626.703-.613.377.014.672.309.66.66l-.694 19.917z"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/memory-user-john-lost.svg diff --git a/1-js/04-object-basics/03-garbage-collection/memory-user-john.svg b/1-js/04-object-basics/03-garbage-collection/memory-user-john.svg new file mode 100644 index 000000000..ef75846a7 --- /dev/null +++ b/1-js/04-object-basics/03-garbage-collection/memory-user-john.svg @@ -0,0 +1,29 @@ +<<<<<<< HEAD:1-js/04-object-basics/02-garbage-collection/memory-user-john.svg +<?xml version="1.0" encoding="UTF-8"?> +<svg width="144px" height="150px" viewBox="0 0 144 150" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> + <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> + <title>memory-user-john.svg</title> + <desc>Created with sketchtool.</desc> + <g id="garbage-collection" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> + <g id="memory-user-john.svg"> + <text id="user" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="31" y="69">user</tspan> + </text> + <rect id="Rectangle-4" stroke="#BCA68E" stroke-width="2" fill="#FFF9EB" x="16" y="95" width="118" height="48"></rect> + <text id="name:-"John"" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" fill="#8A704D"> + <tspan x="32" y="132">name: "John"</tspan> + </text> + <text id="Object-3" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="46" y="112">Object</tspan> + </text> + <path id="Line" d="M73.5,67.5 L73.5,49.5 L75.5,49.5 L75.5,67.5 L81.5,67.5 L74.5,81.5 L67.5,67.5 L73.5,67.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <rect id="Rectangle-1" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB" x="27" y="12" width="93" height="26"></rect> + <text id="<global>" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="43" y="29"><global></tspan> + </text> + </g> + </g> +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="144" height="150" viewBox="0 0 144 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="memory-user-john.svg"><text id="user" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="31" y="69">user</tspan></text><path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M16 95h118v48H16z"/><text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="32" y="132">name: "John"</tspan></text><text id="Object-3" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="46" y="112">Object</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M75.5 49.5v18h6l-7 14-7-14h6v-18h2z"/><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M27 12h93v26H27z"/><text id="<global>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="43" y="29"><global></tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/04-object-basics/03-garbage-collection/memory-user-john.svg diff --git a/1-js/04-object-basics/04-object-methods/4-object-property-this/solution.md b/1-js/04-object-basics/04-object-methods/4-object-property-this/solution.md index f5773ec2c..f33c9310e 100644 --- a/1-js/04-object-basics/04-object-methods/4-object-property-this/solution.md +++ b/1-js/04-object-basics/04-object-methods/4-object-property-this/solution.md @@ -7,21 +7,32 @@ function makeUser() { name: "John", ref: this }; -}; +} let user = makeUser(); alert( user.ref.name ); // Error: Cannot read property 'name' of undefined ``` -That's because rules that set `this` do not look at object literals. +That's because rules that set `this` do not look at object definition. Only the moment of call matters. -Here the value of `this` inside `makeUser()` is `undefined`, because it is called as a function, not as a method. +Here the value of `this` inside `makeUser()` is `undefined`, because it is called as a function, not as a method with "dot" syntax. -And the object literal itself has no effect on `this`. The value of `this` is one for the whole function, code blocks and object literals do not affect it. +The value of `this` is one for the whole function, code blocks and object literals do not affect it. So `ref: this` actually takes current `this` of the function. +We can rewrite the function and return the same `this` with `undefined` value: + +```js run +function makeUser(){ + return this; // this time there's no object literal +} + +alert( makeUser().name ); // Error: Cannot read property 'name' of undefined +``` +As you can see the result of `alert( makeUser().name )` is the same as the result of `alert( user.ref.name )` from the previous example. + Here's the opposite case: ```js run @@ -34,7 +45,7 @@ function makeUser() { } */!* }; -}; +} let user = makeUser(); @@ -42,5 +53,3 @@ alert( user.ref().name ); // John ``` Now it works, because `user.ref()` is a method. And the value of `this` is set to the object before dot `.`. - - diff --git a/1-js/04-object-basics/04-object-methods/4-object-property-this/task.md b/1-js/04-object-basics/04-object-methods/4-object-property-this/task.md index 4784b082c..c6f8f9658 100644 --- a/1-js/04-object-basics/04-object-methods/4-object-property-this/task.md +++ b/1-js/04-object-basics/04-object-methods/4-object-property-this/task.md @@ -14,7 +14,7 @@ function makeUser() { name: "John", ref: this }; -}; +} let user = makeUser(); diff --git a/1-js/04-object-basics/04-object-methods/7-calculator/_js.view/test.js b/1-js/04-object-basics/04-object-methods/7-calculator/_js.view/test.js index 1f71eda4c..4decb76dc 100644 --- a/1-js/04-object-basics/04-object-methods/7-calculator/_js.view/test.js +++ b/1-js/04-object-basics/04-object-methods/7-calculator/_js.view/test.js @@ -15,6 +15,11 @@ describe("calculator", function() { afterEach(function() { prompt.restore(); }); + + it('the read get two values and saves them as object properties', function () { + assert.equal(calculator.a, 2); + assert.equal(calculator.b, 3); + }); it("the sum is 5", function() { assert.equal(calculator.sum(), 5); diff --git a/1-js/04-object-basics/04-object-methods/7-calculator/task.md b/1-js/04-object-basics/04-object-methods/7-calculator/task.md index aa22608ec..82d0da030 100644 --- a/1-js/04-object-basics/04-object-methods/7-calculator/task.md +++ b/1-js/04-object-basics/04-object-methods/7-calculator/task.md @@ -6,7 +6,7 @@ importance: 5 Create an object `calculator` with three methods: -- `read()` prompts for two values and saves them as object properties. +- `read()` prompts for two values and saves them as object properties with names `a` and `b` respectively. - `sum()` returns the sum of saved values. - `mul()` multiplies saved values and returns the result. @@ -21,4 +21,3 @@ alert( calculator.mul() ); ``` [demo] - diff --git a/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/solution.js b/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/solution.js index e98fe6410..a35c009cc 100644 --- a/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/solution.js +++ b/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/solution.js @@ -11,5 +11,6 @@ let ladder = { }, showStep: function() { alert(this.step); + return this; } }; \ No newline at end of file diff --git a/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/test.js b/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/test.js index a2b17fcc4..b4f2459b7 100644 --- a/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/test.js +++ b/1-js/04-object-basics/04-object-methods/8-chain-calls/_js.view/test.js @@ -32,6 +32,14 @@ describe('Ladder', function() { it('down().up().up().up() ', function() { assert.equal(ladder.down().up().up().up().step, 2); }); + + it('showStep() should return this', function() { + assert.equal(ladder.showStep(), ladder); + }); + + it('up().up().down().showStep().down().showStep()', function () { + assert.equal(ladder.up().up().down().showStep().down().showStep().step, 0) + }); after(function() { ladder.step = 0; diff --git a/1-js/04-object-basics/04-object-methods/8-chain-calls/solution.md b/1-js/04-object-basics/04-object-methods/8-chain-calls/solution.md index 2b47873fc..f215461dd 100644 --- a/1-js/04-object-basics/04-object-methods/8-chain-calls/solution.md +++ b/1-js/04-object-basics/04-object-methods/8-chain-calls/solution.md @@ -21,9 +21,9 @@ let ladder = { return this; */!* } -} +}; -ladder.up().up().down().up().down().showStep(); // 1 +ladder.up().up().down().showStep().down().showStep(); // shows 1 then 0 ``` We also can write a single call per line. For long chains it's more readable: @@ -33,7 +33,7 @@ ladder .up() .up() .down() - .up() + .showStep() // 1 .down() - .showStep(); // 1 + .showStep(); // 0 ``` diff --git a/1-js/04-object-basics/04-object-methods/8-chain-calls/task.md b/1-js/04-object-basics/04-object-methods/8-chain-calls/task.md index eca9f4e92..7d2ef8c15 100644 --- a/1-js/04-object-basics/04-object-methods/8-chain-calls/task.md +++ b/1-js/04-object-basics/04-object-methods/8-chain-calls/task.md @@ -4,7 +4,7 @@ importance: 2 # Chaining -There's a `ladder` object that allows to go up and down: +There's a `ladder` object that allows you to go up and down: ```js let ladder = { @@ -21,19 +21,21 @@ let ladder = { }; ``` -Now, if we need to make several calls in sequence, can do it like this: +Now, if we need to make several calls in sequence, we can do it like this: ```js ladder.up(); ladder.up(); ladder.down(); ladder.showStep(); // 1 +ladder.down(); +ladder.showStep(); // 0 ``` -Modify the code of `up`, `down` and `showStep` to make the calls chainable, like this: +Modify the code of `up`, `down`, and `showStep` to make the calls chainable, like this: ```js -ladder.up().up().down().showStep(); // 1 +ladder.up().up().down().showStep().down().showStep(); // shows 1 then 0 ``` -Such approach is widely used across JavaScript libraries. +Such an approach is widely used across JavaScript libraries. diff --git a/1-js/04-object-basics/04-object-methods/article.md b/1-js/04-object-basics/04-object-methods/article.md index 0418adee0..cea2b6a70 100644 --- a/1-js/04-object-basics/04-object-methods/article.md +++ b/1-js/04-object-basics/04-object-methods/article.md @@ -15,7 +15,7 @@ Actions are represented in JavaScript by functions in properties. ## Method examples -For the start, let's teach the `user` to say hello: +For a start, let's teach the `user` to say hello: ```js run let user = { @@ -32,11 +32,11 @@ user.sayHi = function() { user.sayHi(); // Hello! ``` -Here we've just used a Function Expression to create the function and assign it to the property `user.sayHi` of the object. +Here we've just used a Function Expression to create a function and assign it to the property `user.sayHi` of the object. -Then we can call it. The user can now speak! +Then we can call it as `user.sayHi()`. The user can now speak! -A function that is the property of an object is called its *method*. +A function that is a property of an object is called its *method*. So, here we've got a method `sayHi` of the object `user`. @@ -51,7 +51,7 @@ let user = { // first, declare function sayHi() { alert("Hello!"); -}; +} // then add as a method user.sayHi = sayHi; @@ -61,9 +61,9 @@ user.sayHi(); // Hello! ``` ```smart header="Object-oriented programming" -When we write our code using objects to represent entities, that's called an [object-oriented programming](https://en.wikipedia.org/wiki/Object-oriented_programming), in short: "OOP". +When we write our code using objects to represent entities, that's called [object-oriented programming](https://en.wikipedia.org/wiki/Object-oriented_programming), in short: "OOP". -OOP is a big thing, an interesting science of its own. How to choose the right entities? How to organize the interaction between them? That's architecture, and there are great books on that topic, like "Design Patterns: Elements of Reusable Object-Oriented Software" by E.Gamma, R.Helm, R.Johnson, J.Vissides or "Object-Oriented Analysis and Design with Applications" by G.Booch, and more. +OOP is a big thing, an interesting science of its own. How to choose the right entities? How to organize the interaction between them? That's architecture, and there are great books on that topic, like "Design Patterns: Elements of Reusable Object-Oriented Software" by E. Gamma, R. Helm, R. Johnson, J. Vissides or "Object-Oriented Analysis and Design with Applications" by G. Booch, and more. ``` ### Method shorthand @@ -72,16 +72,16 @@ There exists a shorter syntax for methods in an object literal: ```js // these objects do the same -let user = { +user = { sayHi: function() { alert("Hello"); } }; // method shorthand looks better, right? -let user = { +user = { *!* - sayHi() { // same as "sayHi: function()" + sayHi() { // same as "sayHi: function(){...}" */!* alert("Hello"); } @@ -90,7 +90,7 @@ let user = { As demonstrated, we can omit `"function"` and just write `sayHi()`. -To tell the truth, the notations are not fully identical. There are subtle differences related to object inheritance (to be covered later), but for now they do not matter. In almost all cases the shorter syntax is preferred. +To tell the truth, the notations are not fully identical. There are subtle differences related to object inheritance (to be covered later), but for now they do not matter. In almost all cases, the shorter syntax is preferred. ## "this" in methods @@ -111,6 +111,7 @@ let user = { sayHi() { *!* + // "this" is the "current object" alert(this.name); */!* } @@ -159,16 +160,18 @@ let user = { let admin = user; user = null; // overwrite to make things obvious -admin.sayHi(); // Whoops! inside sayHi(), the old name is used! error! +*!* +admin.sayHi(); // TypeError: Cannot read property 'name' of null +*/!* ``` If we used `this.name` instead of `user.name` inside the `alert`, then the code would work. ## "this" is not bound -In JavaScript, "this" keyword behaves unlike most other programming languages. First, it can be used in any function. +In JavaScript, keyword `this` behaves unlike most other programming languages. It can be used in any function, even if it's not a method of an object. -There's no syntax error in the code like that: +There's no syntax error in the following example: ```js function sayHi() { @@ -176,9 +179,9 @@ function sayHi() { } ``` -The value of `this` is evaluated during the run-time. And it can be anything. +The value of `this` is evaluated during the run-time, depending on the context. -For instance, the same function may have different "this" when called from different objects: +For instance, here the same function is assigned to two different objects and has different "this" in the calls: ```js run let user = { name: "John" }; @@ -189,7 +192,7 @@ function sayHi() { } *!* -// use the same functions in two objects +// use the same function in two objects user.f = sayHi; admin.f = sayHi; */!* @@ -202,7 +205,10 @@ admin.f(); // Admin (this == admin) admin['f'](); // Admin (dot or square brackets access the method – doesn't matter) ``` -Actually, we can call the function without an object at all: +The rule is simple: if `obj.f()` is called, then `this` is `obj` during the call of `f`. So it's either `user` or `admin` in the example above. + +````smart header="Calling without an object: `this == undefined`" +We can even call the function without an object at all: ```js run function sayHi() { @@ -216,108 +222,19 @@ In this case `this` is `undefined` in strict mode. If we try to access `this.nam In non-strict mode the value of `this` in such case will be the *global object* (`window` in a browser, we'll get to it later in the chapter [](info:global-object)). This is a historical behavior that `"use strict"` fixes. -Please note that usually a call of a function that uses `this` without an object is not normal, but rather a programming mistake. If a function has `this`, then it is usually meant to be called in the context of an object. +Usually such call is a programming error. If there's `this` inside a function, it expects to be called in an object context. +```` ```smart header="The consequences of unbound `this`" If you come from another programming language, then you are probably used to the idea of a "bound `this`", where methods defined in an object always have `this` referencing that object. -In JavaScript `this` is "free", its value is evaluated at call-time and does not depend on where the method was declared, but rather on what's the object "before the dot". +In JavaScript `this` is "free", its value is evaluated at call-time and does not depend on where the method was declared, but rather on what object is "before the dot". -The concept of run-time evaluated `this` has both pluses and minuses. On the one hand, a function can be reused for different objects. On the other hand, greater flexibility opens a place for mistakes. +The concept of run-time evaluated `this` has both pluses and minuses. On the one hand, a function can be reused for different objects. On the other hand, the greater flexibility creates more possibilities for mistakes. -Here our position is not to judge whether this language design decision is good or bad. We'll understand how to work with it, how to get benefits and evade problems. +Here our position is not to judge whether this language design decision is good or bad. We'll understand how to work with it, how to get benefits and avoid problems. ``` -## Internals: Reference Type - -```warn header="In-depth language feature" -This section covers an advanced topic, to understand certain edge-cases better. - -If you want to go on faster, it can be skipped or postponed. -``` - -An intricate method call can lose `this`, for instance: - -```js run -let user = { - name: "John", - hi() { alert(this.name); }, - bye() { alert("Bye"); } -}; - -user.hi(); // John (the simple call works) - -*!* -// now let's call user.hi or user.bye depending on the name -(user.name == "John" ? user.hi : user.bye)(); // Error! -*/!* -``` - -On the last line there is a ternary operator that chooses either `user.hi` or `user.bye`. In this case the result is `user.hi`. - -The method is immediately called with parentheses `()`. But it doesn't work right! - -You can see that the call results in an error, because the value of `"this"` inside the call becomes `undefined`. - -This works (object dot method): -```js -user.hi(); -``` - -This doesn't (evaluated method): -```js -(user.name == "John" ? user.hi : user.bye)(); // Error! -``` - -Why? If we want to understand why it happens, let's get under the hood of how `obj.method()` call works. - -Looking closely, we may notice two operations in `obj.method()` statement: - -1. First, the dot `'.'` retrieves the property `obj.method`. -2. Then parentheses `()` execute it. - -So, how does the information about `this` get passed from the first part to the second one? - -If we put these operations on separate lines, then `this` will be lost for sure: - -```js run -let user = { - name: "John", - hi() { alert(this.name); } -} - -*!* -// split getting and calling the method in two lines -let hi = user.hi; -hi(); // Error, because this is undefined -*/!* -``` - -Here `hi = user.hi` puts the function into the variable, and then on the last line it is completely standalone, and so there's no `this`. - -**To make `user.hi()` calls work, JavaScript uses a trick -- the dot `'.'` returns not a function, but a value of the special [Reference Type](https://tc39.github.io/ecma262/#sec-reference-specification-type).** - -The Reference Type is a "specification type". We can't explicitly use it, but it is used internally by the language. - -The value of Reference Type is a three-value combination `(base, name, strict)`, where: - -- `base` is the object. -- `name` is the property. -- `strict` is true if `use strict` is in effect. - -The result of a property access `user.hi` is not a function, but a value of Reference Type. For `user.hi` in strict mode it is: - -```js -// Reference Type value -(user, "hi", true) -``` - -When parentheses `()` are called on the Reference Type, they receive the full information about the object and its method, and can set the right `this` (`=user` in this case). - -Any other operation like assignment `hi = user.hi` discards the reference type as a whole, takes the value of `user.hi` (a function) and passes it on. So any further operation "loses" `this`. - -So, as the result, the value of `this` is only passed the right way if the function is called directly using a dot `obj.method()` or square brackets `obj['method']()` syntax (they do the same here). Later in this tutorial, we will learn various ways to solve this problem such as [func.bind()](/bind#solution-2-bind). - ## Arrow functions have no "this" Arrow functions are special: they don't have their "own" `this`. If we reference `this` from such a function, it's taken from the outer "normal" function. @@ -347,7 +264,7 @@ That's a special feature of arrow functions, it's useful when we actually do not The value of `this` is defined at run-time. - When a function is declared, it may use `this`, but that `this` has no value until the function is called. -- That function can be copied between objects. +- A function can be copied between objects. - When a function is called in the "method" syntax: `object.method()`, the value of `this` during the call is `object`. Please note that arrow functions are special: they have no `this`. When `this` is accessed inside an arrow function, it is taken from outside. diff --git a/1-js/04-object-basics/05-object-toprimitive/article.md b/1-js/04-object-basics/05-object-toprimitive/article.md deleted file mode 100644 index a44cf4f4d..000000000 --- a/1-js/04-object-basics/05-object-toprimitive/article.md +++ /dev/null @@ -1,236 +0,0 @@ - -# Object to primitive conversion - -What happens when objects are added `obj1 + obj2`, subtracted `obj1 - obj2` or printed using `alert(obj)`? - -There are special methods in objects that do the conversion. - -In the chapter <info:type-conversions> we've seen the rules for numeric, string and boolean conversions of primitives. But we left a gap for objects. Now, as we know about methods and symbols it becomes possible to close it. - -For objects, there's no to-boolean conversion, because all objects are `true` in a boolean context. So there are only string and numeric conversions. - -The numeric conversion happens when we subtract objects or apply mathematical functions. For instance, `Date` objects (to be covered in the chapter <info:date>) can be subtracted, and the result of `date1 - date2` is the time difference between two dates. - -As for the string conversion -- it usually happens when we output an object like `alert(obj)` and in similar contexts. - -## ToPrimitive - -When an object is used in the context where a primitive is required, for instance, in an `alert` or mathematical operations, it's converted to a primitive value using the `ToPrimitive` algorithm ([specification](https://tc39.github.io/ecma262/#sec-toprimitive)). - -That algorithm allows us to customize the conversion using a special object method. - -Depending on the context, the conversion has a so-called "hint". - -There are three variants: - -`"string"` -: When an operation expects a string, for object-to-string conversions, like `alert`: - - ```js - // output - alert(obj); - - // using object as a property key - anotherObj[obj] = 123; - ``` - -`"number"` -: When an operation expects a number, for object-to-number conversions, like maths: - - ```js - // explicit conversion - let num = Number(obj); - - // maths (except binary plus) - let n = +obj; // unary plus - let delta = date1 - date2; - - // less/greater comparison - let greater = user1 > user2; - ``` - -`"default"` -: Occurs in rare cases when the operator is "not sure" what type to expect. - - For instance, binary plus `+` can work both with strings (concatenates them) and numbers (adds them), so both strings and numbers would do. Or when an object is compared using `==` with a string, number or a symbol. - - ```js - // binary plus - let total = car1 + car2; - - // obj == string/number/symbol - if (user == 1) { ... }; - ``` - - The greater/less operator `<>` can work with both strings and numbers too. Still, it uses "number" hint, not "default". That's for historical reasons. - - In practice, all built-in objects except for one case (`Date` object, we'll learn it later) implement `"default"` conversion the same way as `"number"`. And probably we should do the same. - -Please note -- there are only three hints. It's that simple. There is no "boolean" hint (all objects are `true` in boolean context) or anything else. And if we treat `"default"` and `"number"` the same, like most built-ins do, then there are only two conversions. - -**To do the conversion, JavaScript tries to find and call three object methods:** - -1. Call `obj[Symbol.toPrimitive](hint)` if the method exists, -2. Otherwise if hint is `"string"` - - try `obj.toString()` and `obj.valueOf()`, whatever exists. -3. Otherwise if hint is `"number"` or `"default"` - - try `obj.valueOf()` and `obj.toString()`, whatever exists. - -## Symbol.toPrimitive - -Let's start from the first method. There's a built-in symbol named `Symbol.toPrimitive` that should be used to name the conversion method, like this: - -```js -obj[Symbol.toPrimitive] = function(hint) { - // return a primitive value - // hint = one of "string", "number", "default" -} -``` - -For instance, here `user` object implements it: - -```js run -let user = { - name: "John", - money: 1000, - - [Symbol.toPrimitive](hint) { - alert(`hint: ${hint}`); - return hint == "string" ? `{name: "${this.name}"}` : this.money; - } -}; - -// conversions demo: -alert(user); // hint: string -> {name: "John"} -alert(+user); // hint: number -> 1000 -alert(user + 500); // hint: default -> 1500 -``` - -As we can see from the code, `user` becomes a self-descriptive string or a money amount depending on the conversion. The single method `user[Symbol.toPrimitive]` handles all conversion cases. - - -## toString/valueOf - -Methods `toString` and `valueOf` come from ancient times. They are not symbols (symbols did not exist that long ago), but rather "regular" string-named methods. They provide an alternative "old-style" way to implement the conversion. - -If there's no `Symbol.toPrimitive` then JavaScript tries to find them and try in the order: - -- `toString -> valueOf` for "string" hint. -- `valueOf -> toString` otherwise. - -For instance, here `user` does the same as above using a combination of `toString` and `valueOf`: - -```js run -let user = { - name: "John", - money: 1000, - - // for hint="string" - toString() { - return `{name: "${this.name}"}`; - }, - - // for hint="number" or "default" - valueOf() { - return this.money; - } - -}; - -alert(user); // toString -> {name: "John"} -alert(+user); // valueOf -> 1000 -alert(user + 500); // valueOf -> 1500 -``` - -Often we want a single "catch-all" place to handle all primitive conversions. In this case we can implement `toString` only, like this: - -```js run -let user = { - name: "John", - - toString() { - return this.name; - } -}; - -alert(user); // toString -> John -alert(user + 500); // toString -> John500 -``` - -In the absence of `Symbol.toPrimitive` and `valueOf`, `toString` will handle all primitive conversions. - - -## ToPrimitive and ToString/ToNumber - -The important thing to know about all primitive-conversion methods is that they do not necessarily return the "hinted" primitive. - -There is no control whether `toString()` returns exactly a string, or whether `Symbol.toPrimitive` method returns a number for a hint "number". - -**The only mandatory thing: these methods must return a primitive.** - -An operation that initiated the conversion gets that primitive, and then continues to work with it, applying further conversions if necessary. - -For instance: - -- Mathematical operations (except binary plus) perform `ToNumber` conversion: - - ```js run - let obj = { - toString() { // toString handles all conversions in the absence of other methods - return "2"; - } - }; - - alert(obj * 2); // 4, ToPrimitive gives "2", then it becomes 2 - ``` - -- Binary plus checks the primitive -- if it's a string, then it does concatenation, otherwise it performs `ToNumber` and works with numbers. - - String example: - ```js run - let obj = { - toString() { - return "2"; - } - }; - - alert(obj + 2); // 22 (ToPrimitive returned string => concatenation) - ``` - - Number example: - ```js run - let obj = { - toString() { - return true; - } - }; - - alert(obj + 2); // 3 (ToPrimitive returned boolean, not string => ToNumber) - ``` - -```smart header="Historical notes" -For historical reasons, methods `toString` or `valueOf` *should* return a primitive: if any of them returns an object, then there's no error, but that object is ignored (like if the method didn't exist). - -In contrast, `Symbol.toPrimitive` *must* return a primitive, otherwise, there will be an error. -``` - -## Summary - -The object-to-primitive conversion is called automatically by many built-in functions and operators that expect a primitive as a value. - -There are 3 types (hints) of it: -- `"string"` (for `alert` and other string conversions) -- `"number"` (for maths) -- `"default"` (few operators) - -The specification describes explicitly which operator uses which hint. There are very few operators that "don't know what to expect" and use the `"default"` hint. Usually for built-in objects `"default"` hint is handled the same way as `"number"`, so in practice the last two are often merged together. - -The conversion algorithm is: - -1. Call `obj[Symbol.toPrimitive](hint)` if the method exists, -2. Otherwise if hint is `"string"` - - try `obj.toString()` and `obj.valueOf()`, whatever exists. -3. Otherwise if hint is `"number"` or `"default"` - - try `obj.valueOf()` and `obj.toString()`, whatever exists. - -In practice, it's often enough to implement only `obj.toString()` as a "catch-all" method for all conversions that return a "human-readable" representation of an object, for logging or debugging purposes. diff --git a/1-js/04-object-basics/06-constructor-new/1-two-functions-one-object/task.md b/1-js/04-object-basics/06-constructor-new/1-two-functions-one-object/task.md index 8c1fea8eb..e932a201a 100644 --- a/1-js/04-object-basics/06-constructor-new/1-two-functions-one-object/task.md +++ b/1-js/04-object-basics/06-constructor-new/1-two-functions-one-object/task.md @@ -4,14 +4,14 @@ importance: 2 # Two functions – one object -Is it possible to create functions `A` and `B` such as `new A()==new B()`? +Is it possible to create functions `A` and `B` so that `new A() == new B()`? ```js no-beautify function A() { ... } function B() { ... } -let a = new A; -let b = new B; +let a = new A(); +let b = new B(); alert( a == b ); // true ``` diff --git a/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/_js.view/test.js b/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/_js.view/test.js index 036053927..bba80e5c2 100644 --- a/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/_js.view/test.js +++ b/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/_js.view/test.js @@ -10,6 +10,11 @@ describe("calculator", function() { calculator = new Calculator(); calculator.read(); }); + + it("the read method asks for two values using prompt and remembers them in object properties", function() { + assert.equal(calculator.a, 2); + assert.equal(calculator.b, 3); + }); it("when 2 and 3 are entered, the sum is 5", function() { assert.equal(calculator.sum(), 5); diff --git a/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/task.md b/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/task.md index 60e7c373e..c862bec40 100644 --- a/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/task.md +++ b/1-js/04-object-basics/06-constructor-new/2-calculator-constructor/task.md @@ -6,7 +6,7 @@ importance: 5 Create a constructor function `Calculator` that creates objects with 3 methods: -- `read()` asks for two values using `prompt` and remembers them in object properties. +- `read()` prompts for two values and saves them as object properties with names `a` and `b` respectively. - `sum()` returns the sum of these properties. - `mul()` returns the multiplication product of these properties. diff --git a/1-js/04-object-basics/06-constructor-new/3-accumulator/task.md b/1-js/04-object-basics/06-constructor-new/3-accumulator/task.md index 3362b5b4b..c2c44881e 100644 --- a/1-js/04-object-basics/06-constructor-new/3-accumulator/task.md +++ b/1-js/04-object-basics/06-constructor-new/3-accumulator/task.md @@ -17,8 +17,10 @@ Here's the demo of the code: ```js let accumulator = new Accumulator(1); // initial value 1 + accumulator.read(); // adds the user-entered value accumulator.read(); // adds the user-entered value + alert(accumulator.value); // shows the sum of these values ``` diff --git a/1-js/04-object-basics/06-constructor-new/article.md b/1-js/04-object-basics/06-constructor-new/article.md index eb452f2f3..a335464f1 100644 --- a/1-js/04-object-basics/06-constructor-new/article.md +++ b/1-js/04-object-basics/06-constructor-new/article.md @@ -1,6 +1,6 @@ # Constructor, operator "new" -The regular `{...}` syntax allows to create one object. But often we need to create many similar objects, like multiple users or menu items and so on. +The regular `{...}` syntax allows us to create one object. But often we need to create many similar objects, like multiple users or menu items and so on. That can be done using constructor functions and the `"new"` operator. @@ -27,7 +27,7 @@ alert(user.name); // Jack alert(user.isAdmin); // false ``` -When a function is executed as `new User(...)`, it does the following steps: +When a function is executed with `new`, it does the following steps: 1. A new empty object is created and assigned to `this`. 2. The function body executes. Usually it modifies `this`, adds new properties to it. @@ -51,7 +51,7 @@ function User(name) { } ``` -So the result of `new User("Jack")` is the same object as: +So `let user = new User("Jack")` gives the same result as: ```js let user = { @@ -64,13 +64,14 @@ Now if we want to create other users, we can call `new User("Ann")`, `new User(" That's the main purpose of constructors -- to implement reusable object creation code. -Let's note once again -- technically, any function can be used as a constructor. That is: any function can be run with `new`, and it will execute the algorithm above. The "capital letter first" is a common agreement, to make it clear that a function is to be run with `new`. +Let's note once again -- technically, any function (except arrow functions, as they don't have `this`) can be used as a constructor. It can be run with `new`, and it will execute the algorithm above. The "capital letter first" is a common agreement, to make it clear that a function is to be run with `new`. ````smart header="new function() { ... }" -If we have many lines of code all about creation of a single complex object, we can wrap them in constructor function, like this: +If we have many lines of code all about creation of a single complex object, we can wrap them in an immediately called constructor function, like this: ```js -let user = new function() { +// create a function and immediately call it with new +let user = new function() { this.name = "John"; this.isAdmin = false; @@ -80,10 +81,10 @@ let user = new function() { }; ``` -The constructor can't be called again, because it is not saved anywhere, just created and called. So this trick aims to encapsulate the code that constructs the single object, without future reuse. +This constructor can't be called again, because it is not saved anywhere, just created and called. So this trick aims to encapsulate the code that constructs the single object, without future reuse. ```` -## Dual-syntax constructors: new.target +## Constructor mode test: new.target ```smart header="Advanced stuff" The syntax from this section is rarely used, skip it unless you want to know everything. @@ -91,7 +92,7 @@ The syntax from this section is rarely used, skip it unless you want to know eve Inside a function, we can check whether it was called with `new` or without it, using a special `new.target` property. -It is empty for regular calls and equals the function if called with `new`: +It is undefined for regular calls and equals the function if called with `new`: ```js run function User() { @@ -109,7 +110,9 @@ new User(); // function User { ... } */!* ``` -That can be used to allow both `new` and regular calls to work the same. That is, create the same object: +That can be used inside the function to know whether it was called with `new`, "in constructor mode", or without it, "in regular mode". + +We can also make both `new` and regular calls to do the same, like this: ```js run function User(name) { @@ -134,7 +137,7 @@ Usually, constructors do not have a `return` statement. Their task is to write a But if there is a `return` statement, then the rule is simple: -- If `return` is called with object, then it is returned instead of `this`. +- If `return` is called with an object, then the object is returned instead of `this`. - If `return` is called with a primitive, it's ignored. In other words, `return` with an object returns that object, in all other cases `this` is returned. @@ -146,10 +149,10 @@ function BigUser() { this.name = "John"; - return { name: "Godzilla" }; // <-- returns an object + return { name: "Godzilla" }; // <-- returns this object } -alert( new BigUser().name ); // Godzilla, got that object ^^ +alert( new BigUser().name ); // Godzilla, got that object ``` And here's an example with an empty `return` (or we could place a primitive after it, doesn't matter): @@ -159,10 +162,7 @@ function SmallUser() { this.name = "John"; - return; // finishes the execution, returns this - - // ... - + return; // <-- returns this } alert( new SmallUser().name ); // John @@ -171,7 +171,7 @@ alert( new SmallUser().name ); // John Usually constructors don't have a `return` statement. Here we mention the special behavior with returning objects mainly for the sake of completeness. ````smart header="Omitting parentheses" -By the way, we can omit parentheses after `new`, if it has no arguments: +By the way, we can omit parentheses after `new`: ```js let user = new User; // <-- no parentheses @@ -213,6 +213,8 @@ john = { */ ``` +To create complex objects, there's a more advanced syntax, [classes](info:classes), that we'll cover later. + ## Summary - Constructor functions or, briefly, constructors, are regular functions, but there's a common agreement to name them with capital letter first. diff --git a/1-js/04-object-basics/07-optional-chaining/article.md b/1-js/04-object-basics/07-optional-chaining/article.md new file mode 100644 index 000000000..4c6029423 --- /dev/null +++ b/1-js/04-object-basics/07-optional-chaining/article.md @@ -0,0 +1,233 @@ + +# Optional chaining '?.' + +[recent browser="new"] + +The optional chaining `?.` is a safe way to access nested object properties, even if an intermediate property doesn't exist. + +## The "non-existing property" problem + +If you've just started to read the tutorial and learn JavaScript, maybe the problem hasn't touched you yet, but it's quite common. + +As an example, let's say we have `user` objects that hold the information about our users. + +Most of our users have addresses in `user.address` property, with the street `user.address.street`, but some did not provide them. + +In such case, when we attempt to get `user.address.street`, and the user happens to be without an address, we get an error: + +```js run +let user = {}; // a user without "address" property + +alert(user.address.street); // Error! +``` + +That's the expected result. JavaScript works like this. As `user.address` is `undefined`, an attempt to get `user.address.street` fails with an error. + +In many practical cases we'd prefer to get `undefined` instead of an error here (meaning "no street"). + +...and another example. In Web development, we can get an object that corresponds to a web page element using a special method call, such as `document.querySelector('.elem')`, and it returns `null` when there's no such element. + +```js run +// document.querySelector('.elem') is null if there's no element +let html = document.querySelector('.elem').innerHTML; // error if it's null +``` + +Once again, if the element doesn't exist, we'll get an error accessing `.innerHTML` property of `null`. And in some cases, when the absence of the element is normal, we'd like to avoid the error and just accept `html = null` as the result. + +How can we do this? + +The obvious solution would be to check the value using `if` or the conditional operator `?`, before accessing its property, like this: + +```js +let user = {}; + +alert(user.address ? user.address.street : undefined); +``` + +It works, there's no error... But it's quite inelegant. As you can see, the `"user.address"` appears twice in the code. + +Here's how the same would look for `document.querySelector`: + +```js run +let html = document.querySelector('.elem') ? document.querySelector('.elem').innerHTML : null; +``` + +We can see that the element search `document.querySelector('.elem')` is actually called twice here. Not good. + +For more deeply nested properties, it becomes even uglier, as more repetitions are required. + +E.g. let's get `user.address.street.name` in a similar fashion. + +```js +let user = {}; // user has no address + +alert(user.address ? user.address.street ? user.address.street.name : null : null); +``` + +That's just awful, one may even have problems understanding such code. + +There's a little better way to write it, using the `&&` operator: + +```js run +let user = {}; // user has no address + +alert( user.address && user.address.street && user.address.street.name ); // undefined (no error) +``` + +AND'ing the whole path to the property ensures that all components exist (if not, the evaluation stops), but also isn't ideal. + +As you can see, property names are still duplicated in the code. E.g. in the code above, `user.address` appears three times. + +That's why the optional chaining `?.` was added to the language. To solve this problem once and for all! + +## Optional chaining + +The optional chaining `?.` stops the evaluation if the value before `?.` is `undefined` or `null` and returns `undefined`. + +**Further in this article, for brevity, we'll be saying that something "exists" if it's not `null` and not `undefined`.** + +In other words, `value?.prop`: +- works as `value.prop`, if `value` exists, +- otherwise (when `value` is `undefined/null`) it returns `undefined`. + +Here's the safe way to access `user.address.street` using `?.`: + +```js run +let user = {}; // user has no address + +alert( user?.address?.street ); // undefined (no error) +``` + +The code is short and clean, there's no duplication at all. + +Here's an example with `document.querySelector`: + +```js run +let html = document.querySelector('.elem')?.innerHTML; // will be undefined, if there's no element +``` + +Reading the address with `user?.address` works even if `user` object doesn't exist: + +```js run +let user = null; + +alert( user?.address ); // undefined +alert( user?.address.street ); // undefined +``` + +Please note: the `?.` syntax makes optional the value before it, but not any further. + +E.g. in `user?.address.street.name` the `?.` allows `user` to safely be `null/undefined` (and returns `undefined` in that case), but that's only for `user`. Further properties are accessed in a regular way. If we want some of them to be optional, then we'll need to replace more `.` with `?.`. + +```warn header="Don't overuse the optional chaining" +We should use `?.` only where it's ok that something doesn't exist. + +For example, if according to our code logic `user` object must exist, but `address` is optional, then we should write `user.address?.street`, but not `user?.address?.street`. + +Then, if `user` happens to be undefined, we'll see a programming error about it and fix it. Otherwise, if we overuse `?.`, coding errors can be silenced where not appropriate, and become more difficult to debug. +``` + +````warn header="The variable before `?.` must be declared" +If there's no variable `user` at all, then `user?.anything` triggers an error: + +```js run +// ReferenceError: user is not defined +user?.address; +``` +The variable must be declared (e.g. `let/const/var user` or as a function parameter). The optional chaining works only for declared variables. +```` + +## Short-circuiting + +As it was said before, the `?.` immediately stops ("short-circuits") the evaluation if the left part doesn't exist. + +So, if there are any further function calls or operations to the right of `?.`, they won't be made. + +For instance: + +```js run +let user = null; +let x = 0; + +user?.sayHi(x++); // no "user", so the execution doesn't reach sayHi call and x++ + +alert(x); // 0, value not incremented +``` + +## Other variants: ?.(), ?.[] + +The optional chaining `?.` is not an operator, but a special syntax construct, that also works with functions and square brackets. + +For example, `?.()` is used to call a function that may not exist. + +In the code below, some of our users have `admin` method, and some don't: + +```js run +let userAdmin = { + admin() { + alert("I am admin"); + } +}; + +let userGuest = {}; + +*!* +userAdmin.admin?.(); // I am admin +*/!* + +*!* +userGuest.admin?.(); // nothing happens (no such method) +*/!* +``` + +Here, in both lines we first use the dot (`userAdmin.admin`) to get `admin` property, because we assume that the `user` object exists, so it's safe read from it. + +Then `?.()` checks the left part: if the `admin` function exists, then it runs (that's so for `userAdmin`). Otherwise (for `userGuest`) the evaluation stops without errors. + +The `?.[]` syntax also works, if we'd like to use brackets `[]` to access properties instead of dot `.`. Similar to previous cases, it allows to safely read a property from an object that may not exist. + +```js run +let key = "firstName"; + +let user1 = { + firstName: "John" +}; + +let user2 = null; + +alert( user1?.[key] ); // John +alert( user2?.[key] ); // undefined +``` + +Also we can use `?.` with `delete`: + +```js run +delete user?.name; // delete user.name if user exists +``` + +````warn header="We can use `?.` for safe reading and deleting, but not writing" +The optional chaining `?.` has no use on the left side of an assignment. + +For example: +```js run +let user = null; + +user?.name = "John"; // Error, doesn't work +// because it evaluates to: undefined = "John" +``` + +```` + +## Summary + +The optional chaining `?.` syntax has three forms: + +1. `obj?.prop` -- returns `obj.prop` if `obj` exists, otherwise `undefined`. +2. `obj?.[prop]` -- returns `obj[prop]` if `obj` exists, otherwise `undefined`. +3. `obj.method?.()` -- calls `obj.method()` if `obj.method` exists, otherwise returns `undefined`. + +As we can see, all of them are straightforward and simple to use. The `?.` checks the left part for `null/undefined` and allows the evaluation to proceed if it's not so. + +A chain of `?.` allows to safely access nested properties. + +Still, we should apply `?.` carefully, only where it's acceptable, according to our code logic, that the left part doesn't exist. So that it won't hide programming errors from us, if they occur. diff --git a/1-js/04-object-basics/03-symbol/article.md b/1-js/04-object-basics/08-symbol/article.md similarity index 59% rename from 1-js/04-object-basics/03-symbol/article.md rename to 1-js/04-object-basics/08-symbol/article.md index 8323d6643..10a98af0a 100644 --- a/1-js/04-object-basics/03-symbol/article.md +++ b/1-js/04-object-basics/08-symbol/article.md @@ -1,29 +1,35 @@ # Symbol type -By specification, object property keys may be either of string type, or of symbol type. Not numbers, not booleans, only strings or symbols, these two types. +By specification, only two primitive types may serve as object property keys: -Till now we've only seen strings. Now let's see the advantages that symbols can give us. +- string type, or +- symbol type. + +Otherwise, if one uses another type, such as number, it's autoconverted to string. So that `obj[1]` is the same as `obj["1"]`, and `obj[true]` is the same as `obj["true"]`. + +Until now we've been using only strings. + +Now let's explore symbols, see what they can do for us. ## Symbols -"Symbol" value represents a unique identifier. +A "symbol" represents a unique identifier. A value of this type can be created using `Symbol()`: ```js -// id is a new symbol let id = Symbol(); ``` -We can also give symbol a description (also called a symbol name), mostly useful for debugging purposes: +Upon creation, we can give symbols a description (also called a symbol name), mostly useful for debugging purposes: -```js run +```js // id is a symbol with the description "id" let id = Symbol("id"); ``` -Symbols are guaranteed to be unique. Even if we create many symbols with the same description, they are different values. The description is just a label that doesn't affect anything. +Symbols are guaranteed to be unique. Even if we create many symbols with exactly the same description, they are different values. The description is just a label that doesn't affect anything. For instance, here are two symbols with the same description -- they are not equal: @@ -38,6 +44,8 @@ alert(id1 == id2); // false If you are familiar with Ruby or another language that also has some sort of "symbols" -- please don't be misguided. JavaScript symbols are different. +So, to summarize, a symbol is a "primitive unique value" with an optional description. Let's see where we can use them. + ````warn header="Symbols don't auto-convert to a string" Most values in JavaScript support implicit conversion to a string. For instance, we can `alert` almost any value, and it will work. Symbols are special. They don't auto-convert. @@ -50,9 +58,10 @@ alert(id); // TypeError: Cannot convert a Symbol value to a string */!* ``` -That's a "language guard" against messing up, because strings and symbols are fundamentally different and should not occasionally convert one into another. +That's a "language guard" against messing up, because strings and symbols are fundamentally different and should not accidentally convert one into another. + +If we really want to show a symbol, we need to explicitly call `.toString()` on it, like here: -If we really want to show a symbol, we need to call `.toString()` on it, like here: ```js run let id = Symbol("id"); *!* @@ -60,7 +69,8 @@ alert(id.toString()); // Symbol(id), now it works */!* ``` -Or get `symbol.description` property to get the description only: +Or get `symbol.description` property to show the description only: + ```js run let id = Symbol("id"); *!* @@ -72,23 +82,30 @@ alert(id.description); // id ## "Hidden" properties -Symbols allow us to create "hidden" properties of an object, that no other part of code can occasionally access or overwrite. -For instance, if we want to store an "identifier" for the object `user`, we can use a symbol as a key for it: +Symbols allow us to create "hidden" properties of an object, that no other part of code can accidentally access or overwrite. + +For instance, if we're working with `user` objects, that belong to a third-party code. We'd like to add identifiers to them. + +Let's use a symbol key for it: ```js run -let user = { name: "John" }; +let user = { // belongs to another code + name: "John" +}; + let id = Symbol("id"); -user[id] = "ID Value"; +user[id] = 1; + alert( user[id] ); // we can access the data using the symbol as the key ``` What's the benefit of using `Symbol("id")` over a string `"id"`? -Let's make the example a bit deeper to see that. +As `user` objects belong to another codebase, it's unsafe to add fields to them, since we might affect pre-defined behavior in that other codebase. However, symbols cannot be accessed accidentally. The third-party code won't be aware of newly defined symbols, so it's safe to add symbols to the `user` objects. -Imagine that another script wants to have its own "id" property inside `user`, for its own purposes. That may be another JavaScript library, so the scripts are completely unaware of each other. +Also, imagine that another script wants to have its own identifier inside `user`, for its own purposes. Then that script can create its own `Symbol("id")`, like this: @@ -99,25 +116,25 @@ let id = Symbol("id"); user[id] = "Their id value"; ``` -There will be no conflict, because symbols are always different, even if they have the same name. +There will be no conflict between our and their identifiers, because symbols are always different, even if they have the same name. -Now note that if we used a string `"id"` instead of a symbol for the same purpose, then there *would* be a conflict: +...But if we used a string `"id"` instead of a symbol for the same purpose, then there *would* be a conflict: -```js run +```js let user = { name: "John" }; -// our script uses "id" property -user.id = "ID Value"; +// Our script uses "id" property +user.id = "Our id value"; -// ...if later another script the uses "id" for its purposes... +// ...Another script also wants "id" for its purposes... user.id = "Their id value" -// boom! overwritten! it did not mean to harm the colleague, but did it! +// Boom! overwritten by another script! ``` -### Symbols in a literal +### Symbols in an object literal -If we want to use a symbol in an object literal, we need square brackets. +If we want to use a symbol in an object literal `{...}`, we need square brackets around it. Like this: @@ -127,7 +144,7 @@ let id = Symbol("id"); let user = { name: "John", *!* - [id]: 123 // not just "id: 123" + [id]: 123 // not "id": 123 */!* }; ``` @@ -152,10 +169,10 @@ for (let key in user) alert(key); // name, age (no symbols) */!* // the direct access by the symbol works -alert( "Direct: " + user[id] ); +alert( "Direct: " + user[id] ); // Direct: 123 ``` -That's a part of the general "hiding" concept. If another script or a library loops over our object, it won't unexpectedly access a symbolic property. +[Object.keys(user)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/keys) also ignores them. That's a part of the general "hiding symbolic properties" principle. If another script or a library loops over our object, it won't unexpectedly access a symbolic property. In contrast, [Object.assign](mdn:js/Object/assign) copies both string and symbol properties: @@ -172,31 +189,13 @@ alert( clone[id] ); // 123 There's no paradox here. That's by design. The idea is that when we clone an object or merge objects, we usually want *all* properties to be copied (including symbols like `id`). -````smart header="Property keys of other types are coerced to strings" -We can only use strings or symbols as keys in objects. Other types are converted to strings. - -For instance, a number `0` becomes a string `"0"` when used as a property key: - -```js run -let obj = { - 0: "test" // same as "0": "test" -}; - -// both alerts access the same property (the number 0 is converted to string "0") -alert( obj["0"] ); // test -alert( obj[0] ); // test (same property) -``` -```` - ## Global symbols -As we've seen, usually all symbols are different, even if they have the same names. But sometimes we want same-named symbols to be same entities. - -For instance, different parts of our application want to access symbol `"id"` meaning exactly the same property. +As we've seen, usually all symbols are different, even if they have the same name. But sometimes we want same-named symbols to be same entities. For instance, different parts of our application want to access symbol `"id"` meaning exactly the same property. To achieve that, there exists a *global symbol registry*. We can create symbols in it and access them later, and it guarantees that repeated accesses by the same name return exactly the same symbol. -In order to create or read a symbol in the registry, use `Symbol.for(key)`. +In order to read (create if absent) a symbol from the registry, use `Symbol.for(key)`. That call checks the global registry, and if there's a symbol described as `key`, then returns it, otherwise creates a new symbol `Symbol(key)` and stores it in the registry by the given `key`. @@ -206,7 +205,7 @@ For instance: // read from the global registry let id = Symbol.for("id"); // if the symbol did not exist, it is created -// read it again +// read it again (maybe from another part of the code) let idAgain = Symbol.for("id"); // the same symbol @@ -218,32 +217,39 @@ Symbols inside the registry are called *global symbols*. If we want an applicati ```smart header="That sounds like Ruby" In some programming languages, like Ruby, there's a single symbol per name. -In JavaScript, as we can see, that's right for global symbols. +In JavaScript, as we can see, that's true for global symbols. ``` ### Symbol.keyFor -For global symbols, not only `Symbol.for(key)` returns a symbol by name, but there's a reverse call: `Symbol.keyFor(sym)`, that does the reverse: returns a name by a global symbol. +We have seen that for global symbols, `Symbol.for(key)` returns a symbol by name. To do the opposite -- return a name by global symbol -- we can use: `Symbol.keyFor(sym)`: For instance: ```js run +// get symbol by name let sym = Symbol.for("name"); let sym2 = Symbol.for("id"); -// get name from symbol +// get name by symbol alert( Symbol.keyFor(sym) ); // name alert( Symbol.keyFor(sym2) ); // id ``` -The `Symbol.keyFor` internally uses the global symbol registry to look up the key for the symbol. So it doesn't work for non-global symbols. If the symbol is not global, it won't be able to find it and return `undefined`. +The `Symbol.keyFor` internally uses the global symbol registry to look up the key for the symbol. So it doesn't work for non-global symbols. If the symbol is not global, it won't be able to find it and returns `undefined`. + +That said, all symbols have the `description` property. For instance: ```js run -alert( Symbol.keyFor(Symbol.for("name")) ); // name, global symbol +let globalSymbol = Symbol.for("name"); +let localSymbol = Symbol("name"); + +alert( Symbol.keyFor(globalSymbol) ); // name, global symbol +alert( Symbol.keyFor(localSymbol) ); // undefined, not global -alert( Symbol.keyFor(Symbol("name2")) ); // undefined, the argument isn't a global symbol +alert( localSymbol.description ); // name ``` ## System symbols @@ -266,17 +272,18 @@ Other symbols will also become familiar when we study the corresponding language `Symbol` is a primitive type for unique identifiers. -Symbols are created with `Symbol()` call with an optional description. +Symbols are created with `Symbol()` call with an optional description (name). -Symbols are always different values, even if they have the same name. If we want same-named symbols to be equal, then we should use the global registry: `Symbol.for(key)` returns (creates if needed) a global symbol with `key` as the name. Multiple calls of `Symbol.for` return exactly the same symbol. +Symbols are always different values, even if they have the same name. If we want same-named symbols to be equal, then we should use the global registry: `Symbol.for(key)` returns (creates if needed) a global symbol with `key` as the name. Multiple calls of `Symbol.for` with the same `key` return exactly the same symbol. Symbols have two main use cases: 1. "Hidden" object properties. - If we want to add a property into an object that "belongs" to another script or a library, we can create a symbol and use it as a property key. A symbolic property does not appear in `for..in`, so it won't be occasionally listed. Also it won't be accessed directly, because another script does not have our symbol, so it will not occasionally intervene into its actions. + + If we want to add a property into an object that "belongs" to another script or a library, we can create a symbol and use it as a property key. A symbolic property does not appear in `for..in`, so it won't be accidentally processed together with other properties. Also it won't be accessed directly, because another script does not have our symbol. So the property will be protected from accidental use or overwrite. So we can "covertly" hide something into objects that we need, but others should not see, using symbolic properties. 2. There are many system symbols used by JavaScript which are accessible as `Symbol.*`. We can use them to alter some built-in behaviors. For instance, later in the tutorial we'll use `Symbol.iterator` for [iterables](info:iterable), `Symbol.toPrimitive` to setup [object-to-primitive conversion](info:object-toprimitive) and so on. -Technically, symbols are not 100% hidden. There is a built-in method [Object.getOwnPropertySymbols(obj)](mdn:js/Object/getOwnPropertySymbols) that allows us to get all symbols. Also there is a method named [Reflect.ownKeys(obj)](mdn:js/Reflect/ownKeys) that returns *all* keys of an object including symbolic ones. So they are not really hidden. But most libraries, built-in methods and syntax constructs adhere to a common agreement that they are. And the one who explicitly calls the aforementioned methods probably understands well what he's doing. +Technically, symbols are not 100% hidden. There is a built-in method [Object.getOwnPropertySymbols(obj)](mdn:js/Object/getOwnPropertySymbols) that allows us to get all symbols. Also there is a method named [Reflect.ownKeys(obj)](mdn:js/Reflect/ownKeys) that returns *all* keys of an object including symbolic ones. But most libraries, built-in functions and syntax constructs don't use these methods. diff --git a/1-js/04-object-basics/09-object-toprimitive/article.md b/1-js/04-object-basics/09-object-toprimitive/article.md new file mode 100644 index 000000000..fa68da583 --- /dev/null +++ b/1-js/04-object-basics/09-object-toprimitive/article.md @@ -0,0 +1,280 @@ + +# Object to primitive conversion + +What happens when objects are added `obj1 + obj2`, subtracted `obj1 - obj2` or printed using `alert(obj)`? + +JavaScript doesn't allow you to customize how operators work on objects. Unlike some other programming languages, such as Ruby or C++, we can't implement a special object method to handle addition (or other operators). + +In case of such operations, objects are auto-converted to primitives, and then the operation is carried out over these primitives and results in a primitive value. + +That's an important limitation: the result of `obj1 + obj2` (or another math operation) can't be another object! + +E.g. we can't make objects representing vectors or matrices (or achievements or whatever), add them and expect a "summed" object as the result. Such architectural feats are automatically "off the board". + +So, because we can't technically do much here, there's no maths with objects in real projects. When it happens, with rare exceptions, it's because of a coding mistake. + +In this chapter we'll cover how an object converts to primitive and how to customize it. + +We have two purposes: + +1. It will allow us to understand what's going on in case of coding mistakes, when such an operation happened accidentally. +2. There are exceptions, where such operations are possible and look good. E.g. subtracting or comparing dates (`Date` objects). We'll come across them later. + +## Conversion rules + +In the chapter <info:type-conversions> we've seen the rules for numeric, string and boolean conversions of primitives. But we left a gap for objects. Now, as we know about methods and symbols it becomes possible to fill it. + +1. There's no conversion to boolean. All objects are `true` in a boolean context, as simple as that. There exist only numeric and string conversions. +2. The numeric conversion happens when we subtract objects or apply mathematical functions. For instance, `Date` objects (to be covered in the chapter <info:date>) can be subtracted, and the result of `date1 - date2` is the time difference between two dates. +3. As for the string conversion -- it usually happens when we output an object with `alert(obj)` and in similar contexts. + +We can implement string and numeric conversion by ourselves, using special object methods. + +Now let's get into technical details, because it's the only way to cover the topic in-depth. + +## Hints + +How does JavaScript decide which conversion to apply? + +There are three variants of type conversion, that happen in various situations. They're called "hints", as described in the [specification](https://tc39.github.io/ecma262/#sec-toprimitive): + +`"string"` +: For an object-to-string conversion, when we're doing an operation on an object that expects a string, like `alert`: + + ```js + // output + alert(obj); + + // using object as a property key + anotherObj[obj] = 123; + ``` + +`"number"` +: For an object-to-number conversion, like when we're doing maths: + + ```js + // explicit conversion + let num = Number(obj); + + // maths (except binary plus) + let n = +obj; // unary plus + let delta = date1 - date2; + + // less/greater comparison + let greater = user1 > user2; + ``` + + Most built-in mathematical functions also include such conversion. + +`"default"` +: Occurs in rare cases when the operator is "not sure" what type to expect. + + For instance, binary plus `+` can work both with strings (concatenates them) and numbers (adds them). So if a binary plus gets an object as an argument, it uses the `"default"` hint to convert it. + + Also, if an object is compared using `==` with a string, number or a symbol, it's also unclear which conversion should be done, so the `"default"` hint is used. + + ```js + // binary plus uses the "default" hint + let total = obj1 + obj2; + + // obj == number uses the "default" hint + if (user == 1) { ... }; + ``` + + The greater and less comparison operators, such as `<` `>`, can work with both strings and numbers too. Still, they use the `"number"` hint, not `"default"`. That's for historical reasons. + +In practice though, things are a bit simpler. + +All built-in objects except for one case (`Date` object, we'll learn it later) implement `"default"` conversion the same way as `"number"`. And we probably should do the same. + +Still, it's important to know about all 3 hints, soon we'll see why. + +**To do the conversion, JavaScript tries to find and call three object methods:** + +1. Call `obj[Symbol.toPrimitive](hint)` - the method with the symbolic key `Symbol.toPrimitive` (system symbol), if such method exists, +2. Otherwise if hint is `"string"` + - try calling `obj.toString()` or `obj.valueOf()`, whatever exists. +3. Otherwise if hint is `"number"` or `"default"` + - try calling `obj.valueOf()` or `obj.toString()`, whatever exists. + +## Symbol.toPrimitive + +Let's start from the first method. There's a built-in symbol named `Symbol.toPrimitive` that should be used to name the conversion method, like this: + +```js +obj[Symbol.toPrimitive] = function(hint) { + // here goes the code to convert this object to a primitive + // it must return a primitive value + // hint = one of "string", "number", "default" +}; +``` + +If the method `Symbol.toPrimitive` exists, it's used for all hints, and no more methods are needed. + +For instance, here `user` object implements it: + +```js run +let user = { + name: "John", + money: 1000, + + [Symbol.toPrimitive](hint) { + alert(`hint: ${hint}`); + return hint == "string" ? `{name: "${this.name}"}` : this.money; + } +}; + +// conversions demo: +alert(user); // hint: string -> {name: "John"} +alert(+user); // hint: number -> 1000 +alert(user + 500); // hint: default -> 1500 +``` + +As we can see from the code, `user` becomes a self-descriptive string or a money amount, depending on the conversion. The single method `user[Symbol.toPrimitive]` handles all conversion cases. + +## toString/valueOf + +If there's no `Symbol.toPrimitive` then JavaScript tries to find methods `toString` and `valueOf`: + +- For the `"string"` hint: call `toString` method, and if it doesn't exist or if it returns an object instead of a primitive value, then call `valueOf` (so `toString` has the priority for string conversions). +- For other hints: call `valueOf`, and if it doesn't exist or if it returns an object instead of a primitive value, then call `toString` (so `valueOf` has the priority for maths). + +Methods `toString` and `valueOf` come from ancient times. They are not symbols (symbols did not exist that long ago), but rather "regular" string-named methods. They provide an alternative "old-style" way to implement the conversion. + +These methods must return a primitive value. If `toString` or `valueOf` returns an object, then it's ignored (same as if there were no method). + +By default, a plain object has following `toString` and `valueOf` methods: + +- The `toString` method returns a string `"[object Object]"`. +- The `valueOf` method returns the object itself. + +Here's the demo: + +```js run +let user = {name: "John"}; + +alert(user); // [object Object] +alert(user.valueOf() === user); // true +``` + +So if we try to use an object as a string, like in an `alert` or so, then by default we see `[object Object]`. + +The default `valueOf` is mentioned here only for the sake of completeness, to avoid any confusion. As you can see, it returns the object itself, and so is ignored. Don't ask me why, that's for historical reasons. So we can assume it doesn't exist. + +Let's implement these methods to customize the conversion. + +For instance, here `user` does the same as above using a combination of `toString` and `valueOf` instead of `Symbol.toPrimitive`: + +```js run +let user = { + name: "John", + money: 1000, + + // for hint="string" + toString() { + return `{name: "${this.name}"}`; + }, + + // for hint="number" or "default" + valueOf() { + return this.money; + } + +}; + +alert(user); // toString -> {name: "John"} +alert(+user); // valueOf -> 1000 +alert(user + 500); // valueOf -> 1500 +``` + +As we can see, the behavior is the same as the previous example with `Symbol.toPrimitive`. + +Often we want a single "catch-all" place to handle all primitive conversions. In this case, we can implement `toString` only, like this: + +```js run +let user = { + name: "John", + + toString() { + return this.name; + } +}; + +alert(user); // toString -> John +alert(user + 500); // toString -> John500 +``` + +In the absence of `Symbol.toPrimitive` and `valueOf`, `toString` will handle all primitive conversions. + +### A conversion can return any primitive type + +The important thing to know about all primitive-conversion methods is that they do not necessarily return the "hinted" primitive. + +There is no control whether `toString` returns exactly a string, or whether `Symbol.toPrimitive` method returns a number for the hint `"number"`. + +The only mandatory thing: these methods must return a primitive, not an object. + +```smart header="Historical notes" +For historical reasons, if `toString` or `valueOf` returns an object, there's no error, but such value is ignored (like if the method didn't exist). That's because in ancient times there was no good "error" concept in JavaScript. + +In contrast, `Symbol.toPrimitive` is stricter, it *must* return a primitive, otherwise there will be an error. +``` + +## Further conversions + +As we know already, many operators and functions perform type conversions, e.g. multiplication `*` converts operands to numbers. + +If we pass an object as an argument, then there are two stages of calculations: +1. The object is converted to a primitive (using the rules described above). +2. If necessary for further calculations, the resulting primitive is also converted. + +For instance: + +```js run +let obj = { + // toString handles all conversions in the absence of other methods + toString() { + return "2"; + } +}; + +alert(obj * 2); // 4, object converted to primitive "2", then multiplication made it a number +``` + +1. The multiplication `obj * 2` first converts the object to primitive (that's a string `"2"`). +2. Then `"2" * 2` becomes `2 * 2` (the string is converted to number). + +Binary plus will concatenate strings in the same situation, as it gladly accepts a string: + +```js run +let obj = { + toString() { + return "2"; + } +}; + +alert(obj + 2); // "22" ("2" + 2), conversion to primitive returned a string => concatenation +``` + +## Summary + +The object-to-primitive conversion is called automatically by many built-in functions and operators that expect a primitive as a value. + +There are 3 types (hints) of it: +- `"string"` (for `alert` and other operations that need a string) +- `"number"` (for maths) +- `"default"` (few operators, usually objects implement it the same way as `"number"`) + +The specification describes explicitly which operator uses which hint. + +The conversion algorithm is: + +1. Call `obj[Symbol.toPrimitive](hint)` if the method exists, +2. Otherwise if hint is `"string"` + - try calling `obj.toString()` or `obj.valueOf()`, whatever exists. +3. Otherwise if hint is `"number"` or `"default"` + - try calling `obj.valueOf()` or `obj.toString()`, whatever exists. + +All these methods must return a primitive to work (if defined). + +In practice, it's often enough to implement only `obj.toString()` as a "catch-all" method for string conversions that should return a "human-readable" representation of an object, for logging or debugging purposes. diff --git a/1-js/05-data-types/01-primitives-methods/1-string-new-property/solution.md b/1-js/05-data-types/01-primitives-methods/1-string-new-property/solution.md index a169f7769..fd22a4653 100644 --- a/1-js/05-data-types/01-primitives-methods/1-string-new-property/solution.md +++ b/1-js/05-data-types/01-primitives-methods/1-string-new-property/solution.md @@ -6,26 +6,19 @@ let str = "Hello"; str.test = 5; // (*) -alert(str.test); +alert(str.test); ``` -There may be two kinds of result: -1. `undefined` -2. An error. +Depending on whether you have `use strict` or not, the result may be: +1. `undefined` (no strict mode) +2. An error (strict mode). Why? Let's replay what's happening at line `(*)`: 1. When a property of `str` is accessed, a "wrapper object" is created. -2. The operation with the property is carried out on it. So, the object gets the `test` property. -3. The operation finishes and the "wrapper object" disappears. - -So, on the last line, `str` has no trace of the property. A new wrapper object for every object operation on a string. - -Some browsers though may decide to further limit the programmer and disallow to assign properties to primitives at all. That's why in practice we can also see errors at line `(*)`. It's a little bit farther from the specification though. +2. In strict mode, writing into it is an error. +3. Otherwise, the operation with the property is carried on, the object gets the `test` property, but after that the "wrapper object" disappears, so in the last line `str` has no trace of the property. **This example clearly shows that primitives are not objects.** -They just can not store data. - -All property/method operations are performed with the help of temporary objects. - +They can't store additional data. diff --git a/1-js/05-data-types/01-primitives-methods/1-string-new-property/task.md b/1-js/05-data-types/01-primitives-methods/1-string-new-property/task.md index 50c781ea5..208f84cc7 100644 --- a/1-js/05-data-types/01-primitives-methods/1-string-new-property/task.md +++ b/1-js/05-data-types/01-primitives-methods/1-string-new-property/task.md @@ -15,4 +15,4 @@ str.test = 5; alert(str.test); ``` -How do you think, will it work? What will be shown? +What do you think, will it work? What will be shown? diff --git a/1-js/05-data-types/01-primitives-methods/article.md b/1-js/05-data-types/01-primitives-methods/article.md index a2dcceb19..69e7196e9 100644 --- a/1-js/05-data-types/01-primitives-methods/article.md +++ b/1-js/05-data-types/01-primitives-methods/article.md @@ -1,20 +1,18 @@ # Methods of primitives -JavaScript allows us to work with primitives (strings, numbers, etc.) as if they were objects. - -They also provide methods to call as such. We will study those soon, but first we'll see how it works because, of course, primitives are not objects (and here we will make it even clearer). +JavaScript allows us to work with primitives (strings, numbers, etc.) as if they were objects. They also provide methods to call as such. We will study those soon, but first we'll see how it works because, of course, primitives are not objects (and here we will make it even clearer). Let's look at the key distinctions between primitives and objects. A primitive - Is a value of a primitive type. -- There are 6 primitive types: `string`, `number`, `boolean`, `symbol`, `null` and `undefined`. +- There are 7 primitive types: `string`, `number`, `bigint`, `boolean`, `symbol`, `null` and `undefined`. An object - Is capable of storing multiple values as properties. -- Can be created with `{}`, for instance: `{name: "John", age: 30}`. There are other kinds of objects in JavaScript; functions, for example, are objects. +- Can be created with `{}`, for instance: `{name: "John", age: 30}`. There are other kinds of objects in JavaScript: functions, for example, are objects. One of the best things about objects is that we can store a function as one of its properties. @@ -35,24 +33,24 @@ Many built-in objects already exist, such as those that work with dates, errors, But, these features come with a cost! -Objects are "heavier" than primitives. They require additional resources to support the internal machinery. But as properties and methods are very useful in programming, JavaScript engines try to optimize them to reduce the additional burden. +Objects are "heavier" than primitives. They require additional resources to support the internal machinery. ## A primitive as an object Here's the paradox faced by the creator of JavaScript: -- There are many things one would want to do with a primitive like a string or a number. It would be great to access them as methods. +- There are many things one would want to do with a primitive, like a string or a number. It would be great to access them using methods. - Primitives must be as fast and lightweight as possible. The solution looks a little bit awkward, but here it is: 1. Primitives are still primitive. A single value, as desired. 2. The language allows access to methods and properties of strings, numbers, booleans and symbols. -3. When this happens, a special "object wrapper" that provides the extra functionality is created, and then is destroyed. +3. In order for that to work, a special "object wrapper" that provides the extra functionality is created, and then is destroyed. -The "object wrappers" are different for each primitive type and are called: `String`, `Number`, `Boolean` and `Symbol`. Thus, they provide different sets of methods. +The "object wrappers" are different for each primitive type and are called: `String`, `Number`, `Boolean`, `Symbol` and `BigInt`. Thus, they provide different sets of methods. -For instance, there exists a method [str.toUpperCase()](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/String/toUpperCase) that returns a capitalized string. +For instance, there exists a string method [str.toUpperCase()](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/String/toUpperCase) that returns a capitalized `str`. Here's how it works: @@ -84,31 +82,32 @@ We'll see more specific methods in chapters <info:number> and <info:string>. ````warn header="Constructors `String/Number/Boolean` are for internal use only" -Some languages like Java allow us to create "wrapper objects" for primitives explicitly using a syntax like `new Number(1)` or `new Boolean(false)`. +Some languages like Java allow us to explicitly create "wrapper objects" for primitives using a syntax like `new Number(1)` or `new Boolean(false)`. In JavaScript, that's also possible for historical reasons, but highly **unrecommended**. Things will go crazy in several places. For instance: ```js run -alert( typeof 1 ); // "number" +alert( typeof 0 ); // "number" -alert( typeof new Number(1) ); // "object"! +alert( typeof new Number(0) ); // "object"! ``` -And because what follows, `zero`, is an object, the alert will show up: +Objects are always truthy in `if`, so here the alert will show up: ```js run let zero = new Number(0); if (zero) { // zero is true, because it's an object - alert( "zero is truthy?!?" ); + alert( "zero is truthy!?!" ); } ``` -On the other hand, using the same functions `String/Number/Boolean` without `new` is a totally sane and useful thing. They convert a value to the corresponding type: to a string, a number, or a boolean (primitive). +On the other hand, using the same functions `String/Number/Boolean` without `new` is totally fine and useful thing. They convert a value to the corresponding type: to a string, a number, or a boolean (primitive). For example, this is entirely valid: + ```js let num = Number("123"); // convert a string to number ``` diff --git a/1-js/05-data-types/02-number/2-why-rounded-down/solution.md b/1-js/05-data-types/02-number/2-why-rounded-down/solution.md index a17a4671a..4bcd74512 100644 --- a/1-js/05-data-types/02-number/2-why-rounded-down/solution.md +++ b/1-js/05-data-types/02-number/2-why-rounded-down/solution.md @@ -28,6 +28,6 @@ Note that `63.5` has no precision loss at all. That's because the decimal part ` ```js run -alert( Math.round(6.35 * 10) / 10); // 6.35 -> 63.5 -> 64(rounded) -> 6.4 +alert( Math.round(6.35 * 10) / 10 ); // 6.35 -> 63.5 -> 64(rounded) -> 6.4 ``` diff --git a/1-js/05-data-types/02-number/3-repeat-until-number/_js.view/test.js b/1-js/05-data-types/02-number/3-repeat-until-number/_js.view/test.js index 219fa8068..6bd0123db 100644 --- a/1-js/05-data-types/02-number/3-repeat-until-number/_js.view/test.js +++ b/1-js/05-data-types/02-number/3-repeat-until-number/_js.view/test.js @@ -18,7 +18,7 @@ describe("readNumber", function() { assert.strictEqual(readNumber(), 0); }); - it("continues the loop unti meets a number", function() { + it("continues the loop until meets a number", function() { prompt.onCall(0).returns("not a number"); prompt.onCall(1).returns("not a number again"); prompt.onCall(2).returns("1"); @@ -35,4 +35,4 @@ describe("readNumber", function() { assert.isNull(readNumber()); }); -}); \ No newline at end of file +}); diff --git a/1-js/05-data-types/02-number/article.md b/1-js/05-data-types/02-number/article.md index af06706cd..8e41f673d 100644 --- a/1-js/05-data-types/02-number/article.md +++ b/1-js/05-data-types/02-number/article.md @@ -1,8 +1,12 @@ # Numbers -All numbers in JavaScript are stored in 64-bit format [IEEE-754](https://en.wikipedia.org/wiki/IEEE_754-2008_revision), also known as "double precision floating point numbers". +In modern JavaScript, there are two types of numbers: -Let's recap and expand upon what we currently know about them. +1. Regular numbers in JavaScript are stored in 64-bit format [IEEE-754](https://en.wikipedia.org/wiki/IEEE_754), also known as "double precision floating point numbers". These are numbers that we're using most of the time, and we'll talk about them in this chapter. + +2. BigInt numbers represent integers of arbitrary length. They are sometimes needed because a regular integer number can't safely exceed <code>(2<sup>53</sup>-1)</code> or be less than <code>-(2<sup>53</sup>-1)</code>, as we mentioned earlier in the chapter <info:types>. As bigints are used in a few special areas, we devote them to a special chapter <info:bigint>. + +So here we'll talk about regular numbers. Let's expand our knowledge of them. ## More ways to write a number @@ -12,46 +16,56 @@ Imagine we need to write 1 billion. The obvious way is: let billion = 1000000000; ``` -But in real life we usually avoid writing a long string of zeroes as it's easy to mistype. Also, we are lazy. We will usually write something like `"1bn"` for a billion or `"7.3bn"` for 7 billion 300 million. The same is true for most large numbers. +We also can use underscore `_` as the separator: + +```js +let billion = 1_000_000_000; +``` + +Here the underscore `_` plays the role of the "[syntactic sugar](https://en.wikipedia.org/wiki/Syntactic_sugar)", it makes the number more readable. The JavaScript engine simply ignores `_` between digits, so it's exactly the same one billion as above. -In JavaScript, we shorten a number by appending the letter `"e"` to the number and specifying the zeroes count: +In real life though, we try to avoid writing long sequences of zeroes. We're too lazy for that. We'll try to write something like `"1bn"` for a billion or `"7.3bn"` for 7 billion 300 million. The same is true for most large numbers. + +In JavaScript, we can shorten a number by appending the letter `"e"` to it and specifying the zeroes count: ```js run let billion = 1e9; // 1 billion, literally: 1 and 9 zeroes -alert( 7.3e9 ); // 7.3 billions (7,300,000,000) +alert( 7.3e9 ); // 7.3 billions (same as 7300000000 or 7_300_000_000) ``` -In other words, `"e"` multiplies the number by `1` with the given zeroes count. +In other words, `e` multiplies the number by `1` with the given zeroes count. ```js -1e3 = 1 * 1000 -1.23e6 = 1.23 * 1000000 +1e3 === 1 * 1000; // e3 means *1000 +1.23e6 === 1.23 * 1000000; // e6 means *1000000 ``` - -Now let's write something very small. Say, 1 microsecond (one millionth of a second): +Now let's write something very small. Say, 1 microsecond (one-millionth of a second): ```js -let ms = 0.000001; +let mсs = 0.000001; ``` -Just like before, using `"e"` can help. If we'd like to avoid writing the zeroes explicitly, we could say: +Just like before, using `"e"` can help. If we'd like to avoid writing the zeroes explicitly, we could write the same as: ```js -let ms = 1e-6; // six zeroes to the left from 1 +let mcs = 1e-6; // five zeroes to the left from 1 ``` -If we count the zeroes in `0.000001`, there are 6 of them. So naturally it's `1e-6`. +If we count the zeroes in `0.000001`, there are 6 of them. So naturally it's `1e-6`. In other words, a negative number after `"e"` means a division by 1 with the given number of zeroes: ```js // -3 divides by 1 with 3 zeroes -1e-3 = 1 / 1000 (=0.001) +1e-3 === 1 / 1000; // 0.001 // -6 divides by 1 with 6 zeroes -1.23e-6 = 1.23 / 1000000 (=0.00000123) +1.23e-6 === 1.23 / 1000000; // 0.00000123 + +// an example with a bigger number +1234e-2 === 1234 / 100; // 12.34, decimal point moves 2 times ``` ### Hex, binary and octal numbers @@ -89,13 +103,13 @@ alert( num.toString(16) ); // ff alert( num.toString(2) ); // 11111111 ``` -The `base` can vary from `2` to `36`. By default it's `10`. +The `base` can vary from `2` to `36`. By default, it's `10`. Common use cases for this are: - **base=16** is used for hex colors, character encodings etc, digits can be `0..9` or `A..F`. - **base=2** is mostly for debugging bitwise operations, digits can be `0` or `1`. -- **base=36** is the maximum, digits can be `0..9` or `A..Z`. The whole latin alphabet is used to represent a number. A funny, but useful case for `36` is when we need to turn a long numeric identifier into something shorter, for example to make a short url. Can simply represent it in the numeral system with base `36`: +- **base=36** is the maximum, digits can be `0..9` or `A..Z`. The whole Latin alphabet is used to represent a number. A funny, but useful case for `36` is when we need to turn a long numeric identifier into something shorter, for example, to make a short url. Can simply represent it in the numeral system with base `36`: ```js run alert( 123456..toString(36) ); // 2n9c @@ -104,9 +118,10 @@ Common use cases for this are: ```warn header="Two dots to call a method" Please note that two dots in `123456..toString(36)` is not a typo. If we want to call a method directly on a number, like `toString` in the example above, then we need to place two dots `..` after it. -If we placed a single dot: `123456.toString(36)`, then there would be an error, because JavaScript syntax implies the decimal part after the first dot. And if we place one more dot, then JavaScript knows that the decimal part is empty and now goes the method. +If we placed a single dot: `123456.toString(36)`, then there would be an error, because JavaScript syntax implies the decimal part after the first dot. And if we place one more dot, then JavaScript knows that the decimal part is empty and now uses the method. Also could write `(123456).toString(36)`. + ``` ## Rounding @@ -122,7 +137,7 @@ There are several built-in functions for rounding: : Rounds up: `3.1` becomes `4`, and `-1.1` becomes `-1`. `Math.round` -: Rounds to the nearest integer: `3.1` becomes `3`, `3.6` becomes `4` and `-1.1` becomes `-1`. +: Rounds to the nearest integer: `3.1` becomes `3`, `3.6` becomes `4`. In the middle cases `3.5` rounds up to `4`, and `-3.5` rounds up to `-3`. `Math.trunc` (not supported by Internet Explorer) : Removes anything after the decimal point without rounding: `3.1` becomes `3`, `-1.1` becomes `-1`. @@ -132,8 +147,10 @@ Here's the table to summarize the differences between them: | | `Math.floor` | `Math.ceil` | `Math.round` | `Math.trunc` | |---|---------|--------|---------|---------| |`3.1`| `3` | `4` | `3` | `3` | +|`3.5`| `3` | `4` | `4` | `3` | |`3.6`| `3` | `4` | `4` | `3` | |`-1.1`| `-2` | `-1` | `-1` | `-1` | +|`-1.5`| `-2` | `-1` | `-1` | `-1` | |`-1.6`| `-2` | `-1` | `-2` | `-1` | @@ -149,7 +166,7 @@ There are two ways to do so: ```js run let num = 1.23456; - alert( Math.floor(num * 100) / 100 ); // 1.23456 -> 123.456 -> 123 -> 1.23 + alert( Math.round(num * 100) / 100 ); // 1.23456 -> 123.456 -> 123 -> 1.23 ``` 2. The method [toFixed(n)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/toFixed) rounds the number to `n` digits after the point and returns a string representation of the result. @@ -166,20 +183,20 @@ There are two ways to do so: alert( num.toFixed(1) ); // "12.4" ``` - Please note that result of `toFixed` is a string. If the decimal part is shorter than required, zeroes are appended to the end: + Please note that the result of `toFixed` is a string. If the decimal part is shorter than required, zeroes are appended to the end: ```js run let num = 12.34; alert( num.toFixed(5) ); // "12.34000", added zeroes to make exactly 5 digits ``` - We can convert it to a number using the unary plus or a `Number()` call: `+num.toFixed(5)`. + We can convert it to a number using the unary plus or a `Number()` call, e.g. write `+num.toFixed(5)`. ## Imprecise calculations -Internally, a number is represented in 64-bit format [IEEE-754](https://en.wikipedia.org/wiki/IEEE_754-2008_revision), so there are exactly 64 bits to store a number: 52 of them are used to store the digits, 11 of them store the position of the decimal point (they are zero for integer numbers), and 1 bit is for the sign. +Internally, a number is represented in 64-bit format [IEEE-754](https://en.wikipedia.org/wiki/IEEE_754), so there are exactly 64 bits to store a number: 52 of them are used to store the digits, 11 of them store the position of the decimal point, and 1 bit is for the sign. -If a number is too big, it would overflow the 64-bit storage, potentially giving an infinity: +If a number is really huge, it may overflow the 64-bit storage and become a special numeric value `Infinity`: ```js run alert( 1e500 ); // Infinity @@ -187,7 +204,7 @@ alert( 1e500 ); // Infinity What may be a little less obvious, but happens quite often, is the loss of precision. -Consider this (falsy!) test: +Consider this (falsy!) equality test: ```js run alert( 0.1 + 0.2 == 0.3 ); // *!*false*/!* @@ -201,19 +218,25 @@ Strange! What is it then if not `0.3`? alert( 0.1 + 0.2 ); // 0.30000000000000004 ``` -Ouch! There are more consequences than an incorrect comparison here. Imagine you're making an e-shopping site and the visitor puts `$0.10` and `$0.20` goods into their chart. The order total will be `$0.30000000000000004`. That would surprise anyone. +Ouch! Imagine you're making an e-shopping site and the visitor puts `$0.10` and `$0.20` goods into their cart. The order total will be `$0.30000000000000004`. That would surprise anyone. But why does this happen? -A number is stored in memory in its binary form, a sequence of ones and zeroes. But fractions like `0.1`, `0.2` that look simple in the decimal numeric system are actually unending fractions in their binary form. +A number is stored in memory in its binary form, a sequence of bits - ones and zeroes. But fractions like `0.1`, `0.2` that look simple in the decimal numeric system are actually unending fractions in their binary form. -In other words, what is `0.1`? It is one divided by ten `1/10`, one-tenth. In decimal numeral system such numbers are easily representable. Compare it to one-third: `1/3`. It becomes an endless fraction `0.33333(3)`. +```js run +alert(0.1.toString(2)); // 0.0001100110011001100110011001100110011001100110011001101 +alert(0.2.toString(2)); // 0.001100110011001100110011001100110011001100110011001101 +alert((0.1 + 0.2).toString(2)); // 0.0100110011001100110011001100110011001100110011001101 +``` + +What is `0.1`? It is one divided by ten `1/10`, one-tenth. In the decimal numeral system, such numbers are easily representable. Compare it to one-third: `1/3`. It becomes an endless fraction `0.33333(3)`. So, division by powers `10` is guaranteed to work well in the decimal system, but division by `3` is not. For the same reason, in the binary numeral system, the division by powers of `2` is guaranteed to work, but `1/10` becomes an endless binary fraction. There's just no way to store *exactly 0.1* or *exactly 0.2* using the binary system, just like there is no way to store one-third as a decimal fraction. -The numeric format IEEE-754 solves this by rounding to the nearest possible number. These rounding rules normally don't allow us to see that "tiny precision loss", so the number shows up as `0.3`. But beware, the loss still exists. +The numeric format IEEE-754 solves this by rounding to the nearest possible number. These rounding rules normally don't allow us to see that "tiny precision loss", but it exists. We can see this in action: ```js run @@ -227,14 +250,14 @@ That's why `0.1 + 0.2` is not exactly `0.3`. ```smart header="Not only JavaScript" The same issue exists in many other programming languages. -PHP, Java, C, Perl, Ruby give exactly the same result, because they are based on the same numeric format. +PHP, Java, C, Perl, and Ruby give exactly the same result, because they are based on the same numeric format. ``` Can we work around the problem? Sure, the most reliable method is to round the result with the help of a method [toFixed(n)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/toFixed): ```js run let sum = 0.1 + 0.2; -alert( sum.toFixed(2) ); // 0.30 +alert( sum.toFixed(2) ); // "0.30" ``` Please note that `toFixed` always returns a string. It ensures that it has 2 digits after the decimal point. That's actually convenient if we have an e-shopping and need to show `$0.30`. For other cases, we can use the unary plus to coerce it into a number: @@ -251,7 +274,7 @@ alert( (0.1 * 10 + 0.2 * 10) / 10 ); // 0.3 alert( (0.28 * 100 + 0.14 * 100) / 100); // 0.4200000000000001 ``` -So, multiply/divide approach reduces the error, but doesn't remove it totally. +So, the multiply/divide approach reduces the error, but doesn't remove it totally. Sometimes we could try to evade fractions at all. Like if we're dealing with a shop, then we can store prices in cents instead of dollars. But what if we apply a discount of 30%? In practice, totally evading fractions is rarely possible. Just round them to cut "tails" when needed. @@ -271,13 +294,11 @@ JavaScript doesn't trigger an error in such events. It does its best to fit the ```smart header="Two zeroes" Another funny consequence of the internal representation of numbers is the existence of two zeroes: `0` and `-0`. -That's because a sign is represented by a single bit, so every number can be positive or negative, including a zero. +That's because a sign is represented by a single bit, so it can be set or not set for any number including a zero. -In most cases the distinction is unnoticeable, because operators are suited to treat them as the same. +In most cases, the distinction is unnoticeable, because operators are suited to treat them as the same. ``` - - ## Tests: isFinite and isNaN Remember these two special numeric values? @@ -295,7 +316,7 @@ They belong to the type `number`, but are not "normal" numbers, so there are spe alert( isNaN("str") ); // true ``` - But do we need this function? Can't we just use the comparison `=== NaN`? Sorry, but the answer is no. The value `NaN` is unique in that it does not equal anything, including itself: + But do we need this function? Can't we just use the comparison `=== NaN`? Unfortunately not. The value `NaN` is unique in that it does not equal anything, including itself: ```js run alert( NaN === NaN ); // false @@ -319,18 +340,46 @@ let num = +prompt("Enter a number", ''); alert( isFinite(num) ); ``` -Please note that an empty or a space-only string is treated as `0` in all numeric functions including `isFinite`. +Please note that an empty or a space-only string is treated as `0` in all numeric functions including `isFinite`. -```smart header="Compare with `Object.is`" +````smart header="`Number.isNaN` and `Number.isFinite`" +[Number.isNaN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isNaN) and [Number.isFinite](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Number/isFinite) methods are the more "strict" versions of `isNaN` and `isFinite` functions. They do not autoconvert their argument into a number, but check if it belongs to the `number` type instead. -There is a special built-in method [Object.is](mdn:js/Object/is) that compares values like `===`, but is more reliable for two edge cases: +- `Number.isNaN(value)` returns `true` if the argument belongs to the `number` type and it is `NaN`. In any other case, it returns `false`. + + ```js run + alert( Number.isNaN(NaN) ); // true + alert( Number.isNaN("str" / 2) ); // true + + // Note the difference: + alert( Number.isNaN("str") ); // false, because "str" belongs to the string type, not the number type + alert( isNaN("str") ); // true, because isNaN converts string "str" into a number and gets NaN as a result of this conversion + ``` + +- `Number.isFinite(value)` returns `true` if the argument belongs to the `number` type and it is not `NaN/Infinity/-Infinity`. In any other case, it returns `false`. + + ```js run + alert( Number.isFinite(123) ); // true + alert( Number.isFinite(Infinity) ); // false + alert( Number.isFinite(2 / 0) ); // false + + // Note the difference: + alert( Number.isFinite("123") ); // false, because "123" belongs to the string type, not the number type + alert( isFinite("123") ); // true, because isFinite converts string "123" into a number 123 + ``` + +In a way, `Number.isNaN` and `Number.isFinite` are simpler and more straightforward than `isNaN` and `isFinite` functions. In practice though, `isNaN` and `isFinite` are mostly used, as they're shorter to write. +```` + +```smart header="Comparison with `Object.is`" +There is a special built-in method `Object.is` that compares values like `===`, but is more reliable for two edge cases: 1. It works with `NaN`: `Object.is(NaN, NaN) === true`, that's a good thing. -2. Values `0` and `-0` are different: `Object.is(0, -0) === false`, it rarely matters, but these values technically are different. +2. Values `0` and `-0` are different: `Object.is(0, -0) === false`, technically that's correct because internally the number has a sign bit that may be different even if all other bits are zeroes. In all other cases, `Object.is(a, b)` is the same as `a === b`. -This way of comparison is often used in JavaScript specification. When an internal algorithm needs to compare two values for being exactly the same, it uses `Object.is` (internally called [SameValue](https://tc39.github.io/ecma262/#sec-samevalue)). +We mention `Object.is` here, because it's often used in JavaScript specification. When an internal algorithm needs to compare two values for being exactly the same, it uses `Object.is` (internally called [SameValue](https://tc39.github.io/ecma262/#sec-samevalue)). ``` @@ -344,7 +393,7 @@ alert( +"100px" ); // NaN The sole exception is spaces at the beginning or at the end of the string, as they are ignored. -But in real life we often have values in units, like `"100px"` or `"12pt"` in CSS. Also in many countries the currency symbol goes after the amount, so we have `"19€"` and would like to extract a numeric value out of that. +But in real life, we often have values in units, like `"100px"` or `"12pt"` in CSS. Also in many countries, the currency symbol goes after the amount, so we have `"19€"` and would like to extract a numeric value out of that. That's what `parseInt` and `parseFloat` are for. @@ -382,7 +431,7 @@ JavaScript has a built-in [Math](https://developer.mozilla.org/en/docs/Web/JavaS A few examples: `Math.random()` -: Returns a random number from 0 to 1 (not including 1) +: Returns a random number from 0 to 1 (not including 1). ```js run alert( Math.random() ); // 0.1234567894322 @@ -390,8 +439,8 @@ A few examples: alert( Math.random() ); // ... (any random numbers) ``` -`Math.max(a, b, c...)` / `Math.min(a, b, c...)` -: Returns the greatest/smallest from the arbitrary number of arguments. +`Math.max(a, b, c...)` and `Math.min(a, b, c...)` +: Returns the greatest and smallest from the arbitrary number of arguments. ```js run alert( Math.max(3, 5, -10, 0, 1) ); // 5 @@ -399,27 +448,34 @@ A few examples: ``` `Math.pow(n, power)` -: Returns `n` raised the given power +: Returns `n` raised to the given power. ```js run alert( Math.pow(2, 10) ); // 2 in power 10 = 1024 ``` -There are more functions and constants in `Math` object, including trigonometry, which you can find in the [docs for the Math](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Math) object. +There are more functions and constants in `Math` object, including trigonometry, which you can find in the [docs for the Math object](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Math). ## Summary -To write big numbers: +To write numbers with many zeroes: -- Append `"e"` with the zeroes count to the number. Like: `123e6` is `123` with 6 zeroes. -- A negative number after `"e"` causes the number to be divided by 1 with given zeroes. That's for one-millionth or such. +- Append `"e"` with the zeroes count to the number. Like: `123e6` is the same as `123` with 6 zeroes `123000000`. +- A negative number after `"e"` causes the number to be divided by 1 with given zeroes. E.g. `123e-6` means `0.000123` (`123` millionths). For different numeral systems: -- Can write numbers directly in hex (`0x`), octal (`0o`) and binary (`0b`) systems -- `parseInt(str, base)` parses an integer from any numeral system with base: `2 ≤ base ≤ 36`. +- Can write numbers directly in hex (`0x`), octal (`0o`) and binary (`0b`) systems. +- `parseInt(str, base)` parses the string `str` into an integer in numeral system with given `base`, `2 ≤ base ≤ 36`. - `num.toString(base)` converts a number to a string in the numeral system with the given `base`. +For regular number tests: + +- `isNaN(value)` converts its argument to a number and then tests it for being `NaN` +- `Number.isNaN(value)` checks whether its argument belongs to the `number` type, and if so, tests it for being `NaN` +- `isFinite(value)` converts its argument to a number and then tests it for not being `NaN/Infinity/-Infinity` +- `Number.isFinite(value)` checks whether its argument belongs to the `number` type, and if so, tests it for not being `NaN/Infinity/-Infinity` + For converting values like `12pt` and `100px` to a number: - Use `parseInt/parseFloat` for the "soft" conversion, which reads a number from a string and then returns the value they could read before the error. @@ -431,4 +487,4 @@ For fractions: More mathematical functions: -- See the [Math](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Math) object when you need them. The library is very small, but can cover basic needs. +- See the [Math](https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Math) object when you need them. The library is very small but can cover basic needs. diff --git a/1-js/05-data-types/03-string/1-ucfirst/solution.md b/1-js/05-data-types/03-string/1-ucfirst/solution.md index 4809cf123..be5dd2aaf 100644 --- a/1-js/05-data-types/03-string/1-ucfirst/solution.md +++ b/1-js/05-data-types/03-string/1-ucfirst/solution.md @@ -6,16 +6,11 @@ But we can make a new string based on the existing one, with the uppercased firs let newStr = str[0].toUpperCase() + str.slice(1); ``` -There's a small problem though. If `str` is empty, then `str[0]` is undefined, so we'll get an error. +There's a small problem though. If `str` is empty, then `str[0]` is `undefined`, and as `undefined` doesn't have the `toUpperCase()` method, we'll get an error. -There are two variants here: +The easiest way out is to add a test for an empty string, like this: -1. Use `str.charAt(0)`, as it always returns a string (maybe empty). -2. Add a test for an empty string. - -Here's the 2nd variant: - -```js run +```js run demo function ucFirst(str) { if (!str) return str; @@ -24,4 +19,3 @@ function ucFirst(str) { alert( ucFirst("john") ); // John ``` - diff --git a/1-js/05-data-types/03-string/1-ucfirst/task.md b/1-js/05-data-types/03-string/1-ucfirst/task.md index c0e6ecac4..ed8a1e6a7 100644 --- a/1-js/05-data-types/03-string/1-ucfirst/task.md +++ b/1-js/05-data-types/03-string/1-ucfirst/task.md @@ -2,7 +2,7 @@ importance: 5 --- -# Uppercast the first character +# Uppercase the first character Write a function `ucFirst(str)` that returns the string `str` with the uppercased first character, for instance: diff --git a/1-js/05-data-types/03-string/2-check-spam/solution.md b/1-js/05-data-types/03-string/2-check-spam/solution.md index 893c26497..de8dde57d 100644 --- a/1-js/05-data-types/03-string/2-check-spam/solution.md +++ b/1-js/05-data-types/03-string/2-check-spam/solution.md @@ -1,6 +1,6 @@ To make the search case-insensitive, let's bring the string to lower case and then search: -```js run +```js run demo function checkSpam(str) { let lowerStr = str.toLowerCase(); diff --git a/1-js/05-data-types/03-string/2-check-spam/task.md b/1-js/05-data-types/03-string/2-check-spam/task.md index d073adc05..98b5dd8a0 100644 --- a/1-js/05-data-types/03-string/2-check-spam/task.md +++ b/1-js/05-data-types/03-string/2-check-spam/task.md @@ -4,7 +4,7 @@ importance: 5 # Check for spam -Write a function `checkSpam(str)` that returns `true` if `str` contains 'viagra' or 'XXX', otherwise `false. +Write a function `checkSpam(str)` that returns `true` if `str` contains 'viagra' or 'XXX', otherwise `false`. The function must be case-insensitive: diff --git a/1-js/05-data-types/03-string/3-truncate/solution.md b/1-js/05-data-types/03-string/3-truncate/solution.md index 5546c47ee..d51672ae6 100644 --- a/1-js/05-data-types/03-string/3-truncate/solution.md +++ b/1-js/05-data-types/03-string/3-truncate/solution.md @@ -1,6 +1,6 @@ The maximal length must be `maxlength`, so we need to cut it a little shorter, to give space for the ellipsis. -Note that there is actually a single unicode character for an ellipsis. That's not three dots. +Note that there is actually a single Unicode character for an ellipsis. That's not three dots. ```js run demo function truncate(str, maxlength) { diff --git a/1-js/05-data-types/03-string/3-truncate/task.md b/1-js/05-data-types/03-string/3-truncate/task.md index 6382029f4..c99a5f15a 100644 --- a/1-js/05-data-types/03-string/3-truncate/task.md +++ b/1-js/05-data-types/03-string/3-truncate/task.md @@ -11,7 +11,7 @@ The result of the function should be the truncated (if needed) string. For instance: ```js -truncate("What I'd like to tell on this topic is:", 20) = "What I'd like to te…" +truncate("What I'd like to tell on this topic is:", 20) == "What I'd like to te…" -truncate("Hi everyone!", 20) = "Hi everyone!" +truncate("Hi everyone!", 20) == "Hi everyone!" ``` diff --git a/1-js/05-data-types/03-string/article.md b/1-js/05-data-types/03-string/article.md index e748d65f0..60ce2b6f0 100644 --- a/1-js/05-data-types/03-string/article.md +++ b/1-js/05-data-types/03-string/article.md @@ -17,7 +17,7 @@ let double = "double-quoted"; let backticks = `backticks`; ``` -Single and double quotes are essentially the same. Backticks, however, allow us to embed any expression into the string, including function calls: +Single and double quotes are essentially the same. Backticks, however, allow us to embed any expression into the string, by wrapping it in `${…}`: ```js run function sum(a, b) { @@ -39,60 +39,61 @@ let guestList = `Guests: alert(guestList); // a list of guests, multiple lines ``` -If we try to use single or double quotes in the same way, there will be an error: +Looks natural, right? But single or double quotes do not work this way. + +If we use them and try to use multiple lines, there'll be an error: + ```js run -let guestList = "Guests: // Error: Unexpected token ILLEGAL +let guestList = "Guests: // Error: Unexpected token ILLEGAL * John"; ``` -Single and double quotes come from ancient times of language creation when the need for multiline strings was not taken into account. Backticks appeared much later and thus are more versatile. - -Backticks also allow us to specify a "template function" before the first backtick. The syntax is: <code>func`string`</code>. The function `func` is called automatically, receives the string and embedded expressions and can process them. You can read more about it in the [docs](mdn:/JavaScript/Reference/Template_literals#Tagged_template_literals). This is called "tagged templates". This feature makes it easier to wrap strings into custom templating or other functionality, but it is rarely used. +Single and double quotes come from ancient times of language creation, when the need for multiline strings was not taken into account. Backticks appeared much later and thus are more versatile. +Backticks also allow us to specify a "template function" before the first backtick. The syntax is: <code>func`string`</code>. The function `func` is called automatically, receives the string and embedded expressions and can process them. This feature is called "tagged templates", it's rarely seen, but you can read about it in the MDN: [Template literals](mdn:/JavaScript/Reference/Template_literals#Tagged_templates). ## Special characters -It is still possible to create multiline strings with single quotes by using a so-called "newline character", written as `\n`, which denotes a line break: +It is still possible to create multiline strings with single and double quotes by using a so-called "newline character", written as `\n`, which denotes a line break: ```js run let guestList = "Guests:\n * John\n * Pete\n * Mary"; -alert(guestList); // a multiline list of guests +alert(guestList); // a multiline list of guests, same as above ``` -For example, these two lines describe the same: +As a simpler example, these two lines are equal, just written differently: ```js run -alert( "Hello\nWorld" ); // two lines using a "newline symbol" +let str1 = "Hello\nWorld"; // two lines using a "newline symbol" // two lines using a normal newline and backticks -alert( `Hello -World` ); +let str2 = `Hello +World`; + +alert(str1 == str2); // true ``` -There are other, less common "special" characters as well. Here's the list: +There are other, less common special characters: | Character | Description | |-----------|-------------| -|`\b`|Backspace| -|`\f`|Form feed| |`\n`|New line| -|`\r`|Carriage return| +|`\r`|In Windows text files a combination of two characters `\r\n` represents a new break, while on non-Windows OS it's just `\n`. That's for historical reasons, most Windows software also understands `\n`. | +|`\'`, `\"`, <code>\\`</code>|Quotes| +|`\\`|Backslash| |`\t`|Tab| -|`\uNNNN`|A unicode symbol with the hex code `NNNN`, for instance `\u00A9` -- is a unicode for the copyright symbol `©`. It must be exactly 4 hex digits. | -|`\u{NNNNNNNN}`|Some rare characters are encoded with two unicode symbols, taking up to 4 bytes. This long unicode requires braces around it.| +|`\b`, `\f`, `\v`| Backspace, Form Feed, Vertical Tab -- mentioned for completeness, coming from old times, not used nowadays (you can forget them right now). | -Examples with unicode: +As you can see, all special characters start with a backslash character `\`. It is also called an "escape character". + +Because it's so special, if we need to show an actual backslash `\` within the string, we need to double it: ```js run -alert( "\u00A9" ); // © -alert( "\u{20331}" ); // 佫, a rare chinese hieroglyph (long unicode) -alert( "\u{1F60D}" ); // 😍, a smiling face symbol (another long unicode) +alert( `The backslash: \\` ); // The backslash: \ ``` -All special characters start with a backslash character `\`. It is also called an "escape character". - -We would also use it if we want to insert a quote into the string. +So-called "escaped" quotes `\'`, `\"`, <code>\\`</code> are used to insert a quote into the same-quoted string. For instance: @@ -102,25 +103,16 @@ alert( 'I*!*\'*/!*m the Walrus!' ); // *!*I'm*/!* the Walrus! As you can see, we have to prepend the inner quote by the backslash `\'`, because otherwise it would indicate the string end. -Of course, that refers only to the quotes that are same as the enclosing ones. So, as a more elegant solution, we could switch to double quotes or backticks instead: +Of course, only the quotes that are the same as the enclosing ones need to be escaped. So, as a more elegant solution, we could switch to double quotes or backticks instead: ```js run -alert( `I'm the Walrus!` ); // I'm the Walrus! +alert( "I'm the Walrus!" ); // I'm the Walrus! ``` -Note that the backslash `\` serves for the correct reading of the string by JavaScript, then disappears. The in-memory string has no `\`. You can clearly see that in `alert` from the examples above. - -But what if we need to show an actual backslash `\` within the string? - -That's possible, but we need to double it like `\\`: - -```js run -alert( `The backslash: \\` ); // The backslash: \ -``` +Besides these special characters, there's also a special notation for Unicode codes `\u…`, it's rarely used and is covered in the optional chapter about [Unicode](info:unicode). ## String length - The `length` property has the string length: ```js run @@ -132,33 +124,36 @@ Note that `\n` is a single "special" character, so the length is indeed `3`. ```warn header="`length` is a property" People with a background in some other languages sometimes mistype by calling `str.length()` instead of just `str.length`. That doesn't work. -Please note that `str.length` is a numeric property, not a function. There is no need to add parenthesis after it. +Please note that `str.length` is a numeric property, not a function. There is no need to add parenthesis after it. Not `.length()`, but `.length`. ``` ## Accessing characters -To get a character at position `pos`, use square brackets `[pos]` or call the method [str.charAt(pos)](mdn:js/String/charAt). The first character starts from the zero position: +To get a character at position `pos`, use square brackets `[pos]` or call the method [str.at(pos)](mdn:js/String/at). The first character starts from the zero position: ```js run let str = `Hello`; // the first character alert( str[0] ); // H -alert( str.charAt(0) ); // H +alert( str.at(0) ); // H // the last character alert( str[str.length - 1] ); // o +alert( str.at(-1) ); ``` -The square brackets are a modern way of getting a character, while `charAt` exists mostly for historical reasons. +As you can see, the `.at(pos)` method has a benefit of allowing negative position. If `pos` is negative, then it's counted from the end of the string. -The only difference between them is that if no character is found, `[]` returns `undefined`, and `charAt` returns an empty string: +So `.at(-1)` means the last character, and `.at(-2)` is the one before it, etc. + +The square brackets always return `undefined` for negative indexes, for instance: ```js run let str = `Hello`; -alert( str[1000] ); // undefined -alert( str.charAt(1000) ); // '' (an empty string) +alert( str[-2] ); // undefined +alert( str.at(-2) ); // l ``` We can also iterate over characters using `for..of`: @@ -189,7 +184,7 @@ For instance: ```js run let str = 'Hi'; -str = 'h' + str[1]; // replace the string +str = 'h' + str[1]; // replace the string alert( str ); // hi ``` @@ -207,7 +202,7 @@ alert( 'Interface'.toLowerCase() ); // interface Or, if we want a single character lowercased: -```js +```js run alert( 'Interface'[0].toLowerCase() ); // 'i' ``` @@ -232,7 +227,7 @@ alert( str.indexOf('widget') ); // -1, not found, the search is case-sensitive alert( str.indexOf("id") ); // 1, "id" is found at the position 1 (..idget with id) ``` -The optional second parameter allows us to search starting from the given position. +The optional second parameter allows us to start searching from a given position. For instance, the first occurrence of `"id"` is at position `1`. To look for the next occurrence, let's start the search from position `2`: @@ -242,10 +237,8 @@ let str = 'Widget with id'; alert( str.indexOf('id', 2) ) // 12 ``` - If we're interested in all occurrences, we can run `indexOf` in a loop. Every new call is made with the position after the previous match: - ```js run let str = 'As sly as a fox, as strong as an ox'; @@ -305,41 +298,6 @@ if (str.indexOf("Widget") != -1) { } ``` -````smart header="The bitwise NOT trick" -One of the old tricks used here is the [bitwise NOT](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Bitwise_Operators#Bitwise_NOT) `~` operator. It converts the number to a 32-bit integer (removes the decimal part if exists) and then reverses all bits in its binary representation. - -For 32-bit integers the call `~n` means exactly the same as `-(n+1)` (due to IEEE-754 format). - -For instance: - -```js run -alert( ~2 ); // -3, the same as -(2+1) -alert( ~1 ); // -2, the same as -(1+1) -alert( ~0 ); // -1, the same as -(0+1) -*!* -alert( ~-1 ); // 0, the same as -(-1+1) -*/!* -``` - -As we can see, `~n` is zero only if `n == -1`. - -So, the test `if ( ~str.indexOf("...") )` is truthy that the result of `indexOf` is not `-1`. In other words, when there is a match. - -People use it to shorten `indexOf` checks: - -```js run -let str = "Widget"; - -if (~str.indexOf("Widget")) { - alert( 'Found it!' ); // works -} -``` - -It is usually not recommended to use language features in a non-obvious way, but this particular trick is widely used in old code, so we should understand it. - -Just remember: `if (~str.indexOf(...))` reads as "if found". -```` - ### includes, startsWith, endsWith The more modern method [str.includes(substr, pos)](mdn:js/String/includes) returns `true/false` depending on whether `str` contains `substr` within. @@ -355,15 +313,15 @@ alert( "Hello".includes("Bye") ); // false The optional second argument of `str.includes` is the position to start searching from: ```js run -alert( "Midget".includes("id") ); // true -alert( "Midget".includes("id", 3) ); // false, from position 3 there is no "id" +alert( "Widget".includes("id") ); // true +alert( "Widget".includes("id", 3) ); // false, from position 3 there is no "id" ``` The methods [str.startsWith](mdn:js/String/startsWith) and [str.endsWith](mdn:js/String/endsWith) do exactly what they say: ```js run -alert( "Widget".startsWith("Wid") ); // true, "Widget" starts with "Wid" -alert( "Widget".endsWith("get") ); // true, "Widget" ends with "get" +alert( "*!*Wid*/!*get".startsWith("Wid") ); // true, "Widget" starts with "Wid" +alert( "Wid*!*get*/!*".endsWith("get") ); // true, "Widget" ends with "get" ``` ## Getting a substring @@ -385,7 +343,7 @@ There are 3 methods in JavaScript to get a substring: `substring`, `substr` and ```js run let str = "st*!*ringify*/!*"; - alert( str.slice(2) ); // ringify, from the 2nd position till the end + alert( str.slice(2) ); // 'ringify', from the 2nd position till the end ``` Negative values for `start/end` are also possible. They mean the position is counted from the string end: @@ -394,18 +352,16 @@ There are 3 methods in JavaScript to get a substring: `substring`, `substr` and let str = "strin*!*gif*/!*y"; // start at the 4th position from the right, end at the 1st from the right - alert( str.slice(-4, -1) ); // gif + alert( str.slice(-4, -1) ); // 'gif' ``` - `str.substring(start [, end])` -: Returns the part of the string *between* `start` and `end`. +: Returns the part of the string *between* `start` and `end` (not including `end`). - This is almost the same as `slice`, but it allows `start` to be greater than `end`. + This is almost the same as `slice`, but it allows `start` to be greater than `end` (in this case it simply swaps `start` and `end` values). For instance: - ```js run let str = "st*!*ring*/!*ify"; @@ -421,7 +377,6 @@ There are 3 methods in JavaScript to get a substring: `substring`, `substr` and Negative arguments are (unlike slice) not supported, they are treated as `0`. - `str.substr(start [, length])` : Returns the part of the string from `start`, with the given `length`. @@ -429,29 +384,32 @@ There are 3 methods in JavaScript to get a substring: `substring`, `substr` and ```js run let str = "st*!*ring*/!*ify"; - alert( str.substr(2, 4) ); // ring, from the 2nd position get 4 characters + alert( str.substr(2, 4) ); // 'ring', from the 2nd position get 4 characters ``` The first argument may be negative, to count from the end: ```js run let str = "strin*!*gi*/!*fy"; - alert( str.substr(-4, 2) ); // gi, from the 4th position get 2 characters + alert( str.substr(-4, 2) ); // 'gi', from the 4th position get 2 characters ``` + This method resides in the [Annex B](https://tc39.es/ecma262/#sec-string.prototype.substr) of the language specification. It means that only browser-hosted Javascript engines should support it, and it's not recommended to use it. In practice, it's supported everywhere. + Let's recap these methods to avoid any confusion: | method | selects... | negatives | |--------|-----------|-----------| | `slice(start, end)` | from `start` to `end` (not including `end`) | allows negatives | -| `substring(start, end)` | between `start` and `end` | negative values mean `0` | +| `substring(start, end)` | between `start` and `end` (not including `end`)| negative values mean `0` | | `substr(start, length)` | from `start` get `length` characters | allows negative `start` | - ```smart header="Which one to choose?" All of them can do the job. Formally, `substr` has a minor drawback: it is described not in the core JavaScript specification, but in Annex B, which covers browser-only features that exist mainly for historical reasons. So, non-browser environments may fail to support it. But in practice it works everywhere. -The author finds themself using `slice` almost all the time. +Of the other two variants, `slice` is a little bit more flexible, it allows negative arguments and shorter to write. + +So, for practical use it's enough to remember only `slice`. ``` ## Comparing strings @@ -474,17 +432,18 @@ Although, there are some oddities. This may lead to strange results if we sort these country names. Usually people would expect `Zealand` to come after `Österreich` in the list. -To understand what happens, let's review the internal representation of strings in JavaScript. +To understand what happens, we should be aware that strings in Javascript are encoded using [UTF-16](https://en.wikipedia.org/wiki/UTF-16). That is: each character has a corresponding numeric code. -All strings are encoded using [UTF-16](https://en.wikipedia.org/wiki/UTF-16). That is: each character has a corresponding numeric code. There are special methods that allow to get the character for the code and back. +There are special methods that allow to get the character for the code and back: `str.codePointAt(pos)` -: Returns the code for the character at position `pos`: +: Returns a decimal number representing the code for the character at position `pos`: ```js run // different case letters have different codes - alert( "z".codePointAt(0) ); // 122 alert( "Z".codePointAt(0) ); // 90 + alert( "z".codePointAt(0) ); // 122 + alert( "z".codePointAt(0).toString(16) ); // 7a (if we need a hexadecimal value) ``` `String.fromCodePoint(code)` @@ -492,13 +451,7 @@ All strings are encoded using [UTF-16](https://en.wikipedia.org/wiki/UTF-16). Th ```js run alert( String.fromCodePoint(90) ); // Z - ``` - - We can also add unicode characters by their codes using `\u` followed by the hex code: - - ```js run - // 90 is 5a in hexadecimal system - alert( '\u005a' ); // Z + alert( String.fromCodePoint(0x5a) ); // Z (we can also use a hex value as an argument) ``` Now let's see the characters with codes `65..220` (the latin alphabet and a little bit extra) by making a string of them: @@ -510,35 +463,35 @@ for (let i = 65; i <= 220; i++) { str += String.fromCodePoint(i); } alert( str ); +// Output: // ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~ // ¡¢£¤¥¦§¨©ª«¬®¯°±²³´µ¶·¸¹º»¼½¾¿ÀÁÂÃÄÅÆÇÈÉÊËÌÍÎÏÐÑÒÓÔÕÖרÙÚÛÜ ``` -See? Capital characters go first, then a few special ones, then lowercase characters. +See? Capital characters go first, then a few special ones, then lowercase characters, and `Ö` near the end of the output. Now it becomes obvious why `a > Z`. The characters are compared by their numeric code. The greater code means that the character is greater. The code for `a` (97) is greater than the code for `Z` (90). - All lowercase letters go after uppercase letters because their codes are greater. -- Some letters like `Ö` stand apart from the main alphabet. Here, it's code is greater than anything from `a` to `z`. - +- Some letters like `Ö` stand apart from the main alphabet. Here, its code is greater than anything from `a` to `z`. -### Correct comparisons +### Correct comparisons [#correct-comparisons] -The "right" algorithm to do string comparisons is more complex than it may seem, because alphabets are different for different languages. The same-looking letter may be located differently in different alphabets. +The "right" algorithm to do string comparisons is more complex than it may seem, because alphabets are different for different languages. So, the browser needs to know the language to compare. -Luckily, all modern browsers (IE10- requires the additional library [Intl.JS](https://github.com/andyearnshaw/Intl.js/)) support the internationalization standard [ECMA 402](http://www.ecma-international.org/ecma-402/1.0/ECMA-402.pdf). +Luckily, modern browsers support the internationalization standard [ECMA-402](https://www.ecma-international.org/publications-and-standards/standards/ecma-402/). It provides a special method to compare strings in different languages, following their rules. -The call [str.localeCompare(str2)](mdn:js/String/localeCompare): +The call [str.localeCompare(str2)](mdn:js/String/localeCompare) returns an integer indicating whether `str` is less, equal or greater than `str2` according to the language rules: -- Returns `1` if `str` is greater than `str2` according to the language rules. -- Returns `-1` if `str` is less than `str2`. -- Returns `0` if they are equal. +- Returns a negative number if `str` is less than `str2`. +- Returns a positive number if `str` is greater than `str2`. +- Returns `0` if they are equivalent. For instance: @@ -546,120 +499,13 @@ For instance: alert( 'Österreich'.localeCompare('Zealand') ); // -1 ``` -This method actually has two additional arguments specified in [the documentation](mdn:js/String/localeCompare), which allows it to specify the language (by default taken from the environment) and setup additional rules like case sensitivity or should `"a"` and `"á"` be treated as the same etc. - -## Internals, Unicode - -```warn header="Advanced knowledge" -The section goes deeper into string internals. This knowledge will be useful for you if you plan to deal with emoji, rare mathematical or hieroglyphic characters or other rare symbols. - -You can skip the section if you don't plan to support them. -``` - -### Surrogate pairs - -Most symbols have a 2-byte code. Letters in most european languages, numbers, and even most hieroglyphs, have a 2-byte representation. - -But 2 bytes only allow 65536 combinations and that's not enough for every possible symbol. So rare symbols are encoded with a pair of 2-byte characters called "a surrogate pair". - -The length of such symbols is `2`: - -```js run -alert( '𝒳'.length ); // 2, MATHEMATICAL SCRIPT CAPITAL X -alert( '😂'.length ); // 2, FACE WITH TEARS OF JOY -alert( '𩷶'.length ); // 2, a rare chinese hieroglyph -``` - -Note that surrogate pairs did not exist at the time when JavaScript was created, and thus are not correctly processed by the language! - -We actually have a single symbol in each of the strings above, but the `length` shows a length of `2`. - -`String.fromCodePoint` and `str.codePointAt` are few rare methods that deal with surrogate pairs right. They recently appeared in the language. Before them, there were only [String.fromCharCode](mdn:js/String/fromCharCode) and [str.charCodeAt](mdn:js/String/charCodeAt). These methods are actually the same as `fromCodePoint/codePointAt`, but don't work with surrogate pairs. - -But, for instance, getting a symbol can be tricky, because surrogate pairs are treated as two characters: - -```js run -alert( '𝒳'[0] ); // strange symbols... -alert( '𝒳'[1] ); // ...pieces of the surrogate pair -``` - -Note that pieces of the surrogate pair have no meaning without each other. So the alerts in the example above actually display garbage. - -Technically, surrogate pairs are also detectable by their codes: if a character has the code in the interval of `0xd800..0xdbff`, then it is the first part of the surrogate pair. The next character (second part) must have the code in interval `0xdc00..0xdfff`. These intervals are reserved exclusively for surrogate pairs by the standard. - -In the case above: - -```js run -// charCodeAt is not surrogate-pair aware, so it gives codes for parts - -alert( '𝒳'.charCodeAt(0).toString(16) ); // d835, between 0xd800 and 0xdbff -alert( '𝒳'.charCodeAt(1).toString(16) ); // dcb3, between 0xdc00 and 0xdfff -``` - -You will find more ways to deal with surrogate pairs later in the chapter <info:iterable>. There are probably special libraries for that too, but nothing famous enough to suggest here. - -### Diacritical marks and normalization - -In many languages there are symbols that are composed of the base character with a mark above/under it. - -For instance, the letter `a` can be the base character for: `àáâäãåā`. Most common "composite" character have their own code in the UTF-16 table. But not all of them, because there are too many possible combinations. - -To support arbitrary compositions, UTF-16 allows us to use several unicode characters. The base character and one or many "mark" characters that "decorate" it. - -For instance, if we have `S` followed by the special "dot above" character (code `\u0307`), it is shown as Ṡ. - -```js run -alert( 'S\u0307' ); // Ṡ -``` - -If we need an additional mark above the letter (or below it) -- no problem, just add the necessary mark character. - -For instance, if we append a character "dot below" (code `\u0323`), then we'll have "S with dots above and below": `Ṩ`. - -For example: - -```js run -alert( 'S\u0307\u0323' ); // Ṩ -``` - -This provides great flexibility, but also an interesting problem: two characters may visually look the same, but be represented with different unicode compositions. - -For instance: - -```js run -alert( 'S\u0307\u0323' ); // Ṩ, S + dot above + dot below -alert( 'S\u0323\u0307' ); // Ṩ, S + dot below + dot above - -alert( 'S\u0307\u0323' == 'S\u0323\u0307' ); // false -``` - -To solve this, there exists a "unicode normalization" algorithm that brings each string to the single "normal" form. - -It is implemented by [str.normalize()](mdn:js/String/normalize). - -```js run -alert( "S\u0307\u0323".normalize() == "S\u0323\u0307".normalize() ); // true -``` - -It's funny that in our situation `normalize()` actually brings together a sequence of 3 characters to one: `\u1e68` (S with two dots). - -```js run -alert( "S\u0307\u0323".normalize().length ); // 1 - -alert( "S\u0307\u0323".normalize() == "\u1e68" ); // true -``` - -In reality, this is not always the case. The reason being that the symbol `Ṩ` is "common enough", so UTF-16 creators included it in the main table and gave it the code. - -If you want to learn more about normalization rules and variants -- they are described in the appendix of the Unicode standard: [Unicode Normalization Forms](http://www.unicode.org/reports/tr15/), but for most practical purposes the information from this section is enough. - +This method actually has two additional arguments specified in [the documentation](mdn:js/String/localeCompare), which allows it to specify the language (by default taken from the environment, letter order depends on the language) and setup additional rules like case sensitivity or should `"a"` and `"á"` be treated as the same etc. ## Summary -- There are 3 types of quotes. Backticks allow a string to span multiple lines and embed expressions. -- Strings in JavaScript are encoded using UTF-16. -- We can use special characters like `\n` and insert letters by their unicode using `\u...`. -- To get a character, use: `[]`. +- There are 3 types of quotes. Backticks allow a string to span multiple lines and embed expressions `${…}`. +- We can use special characters, such as a line break `\n`. +- To get a character, use: `[]` or `at` method. - To get a substring, use: `slice` or `substring`. - To lowercase/uppercase a string, use: `toLowerCase/toUpperCase`. - To look for a substring, use: `indexOf`, or `includes/startsWith/endsWith` for simple checks. @@ -669,6 +515,8 @@ There are several other helpful methods in strings: - `str.trim()` -- removes ("trims") spaces from the beginning and end of the string. - `str.repeat(n)` -- repeats the string `n` times. -- ...and more. See the [manual](mdn:js/String) for details. +- ...and more to be found in the [manual](mdn:js/String). + +Strings also have methods for doing search/replace with regular expressions. But that's big topic, so it's explained in a separate tutorial section <info:regular-expressions>. -Strings also have methods for doing search/replace with regular expressions. But that topic deserves a separate chapter, so we'll return to that later. +Also, as of now it's important to know that strings are based on Unicode encoding, and hence there're issues with comparisons. There's more about Unicode in the chapter <info:unicode>. diff --git a/1-js/05-data-types/04-array/10-maximal-subarray/solution.md b/1-js/05-data-types/04-array/10-maximal-subarray/solution.md index daadf494b..7e1ca3bde 100644 --- a/1-js/05-data-types/04-array/10-maximal-subarray/solution.md +++ b/1-js/05-data-types/04-array/10-maximal-subarray/solution.md @@ -57,9 +57,9 @@ alert( getMaxSubSum([1, 2, 3]) ); // 6 alert( getMaxSubSum([100, -9, 2, -3, 5]) ); // 100 ``` -The solution has a time complexety of [O(n<sup>2</sup>)](https://en.wikipedia.org/wiki/Big_O_notation). In other words, if we increase the array size 2 times, the algorithm will work 4 times longer. +The solution has a time complexity of [O(n<sup>2</sup>)](https://en.wikipedia.org/wiki/Big_O_notation). In other words, if we increase the array size 2 times, the algorithm will work 4 times longer. -For big arrays (1000, 10000 or more items) such algorithms can lead to a serious sluggishness. +For big arrays (1000, 10000 or more items) such algorithms can lead to serious sluggishness. # Fast solution @@ -91,4 +91,4 @@ alert( getMaxSubSum([-1, -2, -3]) ); // 0 The algorithm requires exactly 1 array pass, so the time complexity is O(n). -You can find more detail information about the algorithm here: [Maximum subarray problem](http://en.wikipedia.org/wiki/Maximum_subarray_problem). If it's still not obvious why that works, then please trace the algorithm on the examples above, see how it works, that's better than any words. +You can find more detailed information about the algorithm here: [Maximum subarray problem](http://en.wikipedia.org/wiki/Maximum_subarray_problem). If it's still not obvious why that works, then please trace the algorithm on the examples above, see how it works, that's better than any words. diff --git a/1-js/05-data-types/04-array/10-maximal-subarray/task.md b/1-js/05-data-types/04-array/10-maximal-subarray/task.md index e63c4e625..f1a1d9f95 100644 --- a/1-js/05-data-types/04-array/10-maximal-subarray/task.md +++ b/1-js/05-data-types/04-array/10-maximal-subarray/task.md @@ -10,15 +10,15 @@ The task is: find the contiguous subarray of `arr` with the maximal sum of items Write the function `getMaxSubSum(arr)` that will return that sum. -For instance: +For instance: ```js -getMaxSubSum([-1, *!*2, 3*/!*, -9]) = 5 (the sum of highlighted items) -getMaxSubSum([*!*2, -1, 2, 3*/!*, -9]) = 6 -getMaxSubSum([-1, 2, 3, -9, *!*11*/!*]) = 11 -getMaxSubSum([-2, -1, *!*1, 2*/!*]) = 3 -getMaxSubSum([*!*100*/!*, -9, 2, -3, 5]) = 100 -getMaxSubSum([*!*1, 2, 3*/!*]) = 6 (take all) +getMaxSubSum([-1, *!*2, 3*/!*, -9]) == 5 (the sum of highlighted items) +getMaxSubSum([*!*2, -1, 2, 3*/!*, -9]) == 6 +getMaxSubSum([-1, 2, 3, -9, *!*11*/!*]) == 11 +getMaxSubSum([-2, -1, *!*1, 2*/!*]) == 3 +getMaxSubSum([*!*100*/!*, -9, 2, -3, 5]) == 100 +getMaxSubSum([*!*1, 2, 3*/!*]) == 6 (take all) ``` If all items are negative, it means that we take none (the subarray is empty), so the sum is zero: diff --git a/1-js/05-data-types/04-array/2-create-array/task.md b/1-js/05-data-types/04-array/2-create-array/task.md index 3e9300793..d4551c79c 100644 --- a/1-js/05-data-types/04-array/2-create-array/task.md +++ b/1-js/05-data-types/04-array/2-create-array/task.md @@ -8,7 +8,7 @@ Let's try 5 array operations. 1. Create an array `styles` with items "Jazz" and "Blues". 2. Append "Rock-n-Roll" to the end. -3. Replace the value in the middle by "Classics". Your code for finding the middle value should work for any arrays with odd length. +3. Replace the value in the middle with "Classics". Your code for finding the middle value should work for any arrays with odd length. 4. Strip off the first value of the array and show it. 5. Prepend `Rap` and `Reggae` to the array. @@ -16,7 +16,7 @@ The array in the process: ```js no-beautify Jazz, Blues -Jazz, Bues, Rock-n-Roll +Jazz, Blues, Rock-n-Roll Jazz, Classics, Rock-n-Roll Classics, Rock-n-Roll Rap, Reggae, Classics, Rock-n-Roll diff --git a/1-js/05-data-types/04-array/3-call-array-this/solution.md b/1-js/05-data-types/04-array/3-call-array-this/solution.md index e994ae078..3cb0317cf 100644 --- a/1-js/05-data-types/04-array/3-call-array-this/solution.md +++ b/1-js/05-data-types/04-array/3-call-array-this/solution.md @@ -9,7 +9,7 @@ arr.push(function() { alert( this ); }) -arr[2](); // "a","b",function +arr[2](); // a,b,function(){...} ``` The array has 3 values: initially it had two, plus the function. diff --git a/1-js/05-data-types/04-array/3-call-array-this/task.md b/1-js/05-data-types/04-array/3-call-array-this/task.md index 340c5feef..f1e13499c 100644 --- a/1-js/05-data-types/04-array/3-call-array-this/task.md +++ b/1-js/05-data-types/04-array/3-call-array-this/task.md @@ -11,7 +11,7 @@ let arr = ["a", "b"]; arr.push(function() { alert( this ); -}) +}); arr[2](); // ? ``` diff --git a/1-js/05-data-types/04-array/article.md b/1-js/05-data-types/04-array/article.md index aead164f5..e71e86a5b 100644 --- a/1-js/05-data-types/04-array/article.md +++ b/1-js/05-data-types/04-array/article.md @@ -1,12 +1,12 @@ -# Arrays +# Arrays Objects allow you to store keyed collections of values. That's fine. -But quite often we find that we need an *ordered collection*, where we have a 1st, a 2nd, a 3rd element and so on. For example, we need that to store a list of something: users, goods, HTML elements etc. +But quite often we find that we need an *ordered collection*, where we have a 1st, a 2nd, a 3rd element and so on. For example, we need that to store a list of something: users, goods, HTML elements etc. It is not convenient to use an object here, because it provides no methods to manage the order of elements. We can’t insert a new property “between” the existing ones. Objects are just not meant for such use. -There exists a special data structure named `Array`, to store ordered collections. +There exists a special data structure named `Array`, to store ordered collections. ## Declaration @@ -81,10 +81,10 @@ arr[3](); // hello ````smart header="Trailing comma" An array, just like an object, may end with a comma: -```js +```js let fruits = [ - "Apple", - "Orange", + "Apple", + "Orange", "Plum"*!*,*/!* ]; ``` @@ -92,10 +92,42 @@ let fruits = [ The "trailing comma" style makes it easier to insert/remove items, because all lines become alike. ```` +## Get last elements with "at" + +[recent browser="new"] + +Let's say we want the last element of the array. + +Some programming languages allow the use of negative indexes for the same purpose, like `fruits[-1]`. + +Although, in JavaScript it won't work. The result will be `undefined`, because the index in square brackets is treated literally. + +We can explicitly calculate the last element index and then access it: `fruits[fruits.length - 1]`. + +```js run +let fruits = ["Apple", "Orange", "Plum"]; + +alert( fruits[fruits.length-1] ); // Plum +``` + +A bit cumbersome, isn't it? We need to write the variable name twice. + +Luckily, there's a shorter syntax: `fruits.at(-1)`: + +```js run +let fruits = ["Apple", "Orange", "Plum"]; + +// same as fruits[fruits.length-1] +alert( fruits.at(-1) ); // Plum +``` + +In other words, `arr.at(i)`: +- is exactly the same as `arr[i]`, if `i >= 0`. +- for negative values of `i`, it steps back from the end of the array. ## Methods pop/push, shift/unshift -A [queue](https://en.wikipedia.org/wiki/Queue_(abstract_data_type)) is one of most common uses of an array. In computer science, this means an ordered collection of elements which supports two operations: +A [queue](https://en.wikipedia.org/wiki/Queue_(abstract_data_type)) is one of the most common uses of an array. In computer science, this means an ordered collection of elements which supports two operations: - `push` appends an element to the end. - `shift` get an element from the beginning, advancing the queue, so that the 2nd element becomes the 1st. @@ -106,7 +138,7 @@ Arrays support both operations. In practice we need it very often. For example, a queue of messages that need to be shown on-screen. -There's another use case for arrays -- the data structure named [stack](https://en.wikipedia.org/wiki/Stack_(abstract_data_type)). +There's another use case for arrays -- the data structure named [stack](https://en.wikipedia.org/wiki/Stack_(abstract_data_type)). It supports two operations: @@ -121,9 +153,9 @@ A stack is usually illustrated as a pack of cards: new cards are added to the to For stacks, the latest pushed item is received first, that's also called LIFO (Last-In-First-Out) principle. For queues, we have FIFO (First-In-First-Out). -Arrays in JavaScript can work both as a queue and as a stack. They allow you to add/remove elements both to/from the beginning or the end. +Arrays in JavaScript can work both as a queue and as a stack. They allow you to add/remove elements, both to/from the beginning or the end. -In computer science the data structure that allows it is called [deque](https://en.wikipedia.org/wiki/Double-ended_queue). +In computer science, the data structure that allows this, is called [deque](https://en.wikipedia.org/wiki/Double-ended_queue). **Methods that work with the end of the array:** @@ -138,6 +170,8 @@ In computer science the data structure that allows it is called [deque](https:// alert( fruits ); // Apple, Orange ``` + Both `fruits.pop()` and `fruits.at(-1)` return the last element of the array, but `fruits.pop()` also modifies the array by removing it. + `push` : Append the element to the end of the array: @@ -156,7 +190,7 @@ In computer science the data structure that allows it is called [deque](https:// `shift` : Extracts the first element of the array and returns it: - ```js + ```js run let fruits = ["Apple", "Orange", "Pear"]; alert( fruits.shift() ); // remove Apple and alert it @@ -167,7 +201,7 @@ In computer science the data structure that allows it is called [deque](https:// `unshift` : Add the element to the beginning of the array: - ```js + ```js run let fruits = ["Orange", "Pear"]; fruits.unshift('Apple'); @@ -189,11 +223,11 @@ alert( fruits ); ## Internals -An array is a special kind of object. The square brackets used to access a property `arr[0]` actually come from the object syntax. Numbers are used as keys. +An array is a special kind of object. The square brackets used to access a property `arr[0]` actually come from the object syntax. That's essentially the same as `obj[key]`, where `arr` is the object, while numbers are used as keys. They extend objects providing special methods to work with ordered collections of data and also the `length` property. But at the core it's still an object. -Remember, there are only 7 basic types in JavaScript. Array is an object and thus behaves like an object. +Remember, there are only eight basic data types in JavaScript (see the [Data types](info:types) chapter for more info). Array is an object and thus behaves like an object. For instance, it is copied by reference: @@ -203,13 +237,13 @@ let fruits = ["Banana"] let arr = fruits; // copy by reference (two variables reference the same array) alert( arr === fruits ); // true - + arr.push("Pear"); // modify the array by reference alert( fruits ); // Banana, Pear - 2 items now ``` -...But what makes arrays really special is their internal representation. The engine tries to store its elements in the contiguous memory area, one after another, just as depicted on the illustrations in this chapter, and there are other optimizations as well, to make arrays work really fast. +...But what makes arrays really special is their internal representation. The engine tries to store its elements in the contiguous memory area, one after another, just as depicted on the illustrations in this chapter, and there are other optimizations as well, to make arrays work really fast. But they all break if we quit working with an array as with an "ordered collection" and start working with it as if it were a regular object. @@ -229,7 +263,7 @@ But the engine will see that we're working with the array as with a regular obje The ways to misuse an array: -- Add a non-numeric property like `arr.test = 5`. +- Add a non-numeric property like `arr.test = 5`. - Make holes, like: add `arr[0]` and then `arr[1000]` (and nothing between them). - Fill the array in the reverse order, like `arr[1000]`, `arr[999]` and so on. @@ -247,7 +281,7 @@ Why is it faster to work with the end of an array than with its beginning? Let's fruits.shift(); // take 1 element from the start ``` -It's not enough to take and remove the element with the number `0`. Other elements need to be renumbered as well. +It's not enough to take and remove the element with the index `0`. Other elements need to be renumbered as well. The `shift` operation must do 3 things: @@ -296,7 +330,7 @@ let fruits = ["Apple", "Orange", "Plum"]; // iterates over array elements for (let fruit of fruits) { - alert( fruit ); + alert( fruit ); } ``` @@ -320,7 +354,7 @@ But that's actually a bad idea. There are potential problems with it: There are so-called "array-like" objects in the browser and in other environments, that *look like arrays*. That is, they have `length` and indexes properties, but they may also have other non-numeric properties and methods, which we usually don't need. The `for..in` loop will list them though. So if we need to work with array-like objects, then these "extra" properties can become a problem. -2. The `for..in` loop is optimized for generic objects, not arrays, and thus is 10-100 times slower. Of course, it's still very fast. The speedup may only matter in bottlenecks or seem irrelevant. But still we should be aware of the difference. +2. The `for..in` loop is optimized for generic objects, not arrays, and thus is 10-100 times slower. Of course, it's still very fast. The speedup may only matter in bottlenecks. But still we should be aware of the difference. Generally, we shouldn't use `for..in` for arrays. @@ -338,7 +372,7 @@ fruits[123] = "Apple"; alert( fruits.length ); // 124 ``` -Note that we usually don't use arrays like that. +Note that we usually don't use arrays like that. Another interesting thing about the `length` property is that it's writable. @@ -365,11 +399,11 @@ There is one more syntax to create an array: let arr = *!*new Array*/!*("Apple", "Pear", "etc"); ``` -It's rarely used, because square brackets `[]` are shorter. Also there's a tricky feature with it. +It's rarely used, because square brackets `[]` are shorter. Also, there's a tricky feature with it. If `new Array` is called with a single argument which is a number, then it creates an array *without items, but with the given length*. -Let's see how one can shoot themself in the foot: +Let's see how one can shoot themselves in the foot: ```js run let arr = new Array(2); // will it create an array of [2] ? @@ -379,13 +413,11 @@ alert( arr[0] ); // undefined! no elements. alert( arr.length ); // length 2 ``` -In the code above, `new Array(number)` has all elements `undefined`. - -To evade such surprises, we usually use square brackets, unless we really know what we're doing. +To avoid such surprises, we usually use square brackets, unless we really know what we're doing. ## Multidimensional arrays -Arrays can have items that are also arrays. We can use it for multidimensional arrays, to store matrices: +Arrays can have items that are also arrays. We can use it for multidimensional arrays, for example to store matrices: ```js run let matrix = [ @@ -394,7 +426,7 @@ let matrix = [ [7, 8, 9] ]; -alert( matrix[1][1] ); // the central element +alert( matrix[0][1] ); // 2, the second value of the first inner array ``` ## toString @@ -429,36 +461,91 @@ alert( "1" + 1 ); // "11" alert( "1,2" + 1 ); // "1,21" ``` +## Don't compare arrays with == + +Arrays in JavaScript, unlike some other programming languages, shouldn't be compared with operator `==`. + +This operator has no special treatment for arrays, it works with them as with any objects. + +Let's recall the rules: + +- Two objects are equal `==` only if they're references to the same object. +- If one of the arguments of `==` is an object, and the other one is a primitive, then the object gets converted to primitive, as explained in the chapter <info:object-toprimitive>. +- ...With an exception of `null` and `undefined` that equal `==` each other and nothing else. + +The strict comparison `===` is even simpler, as it doesn't convert types. + +So, if we compare arrays with `==`, they are never the same, unless we compare two variables that reference exactly the same array. + +For example: +```js run +alert( [] == [] ); // false +alert( [0] == [0] ); // false +``` + +These arrays are technically different objects. So they aren't equal. The `==` operator doesn't do item-by-item comparison. + +Comparison with primitives may give seemingly strange results as well: + +```js run +alert( 0 == [] ); // true + +alert('0' == [] ); // false +``` + +Here, in both cases, we compare a primitive with an array object. So the array `[]` gets converted to primitive for the purpose of comparison and becomes an empty string `''`. + +Then the comparison process goes on with the primitives, as described in the chapter <info:type-conversions>: + +```js run +// after [] was converted to '' +alert( 0 == '' ); // true, as '' becomes converted to number 0 + +alert('0' == '' ); // false, no type conversion, different strings +``` + +So, how to compare arrays? + +That's simple: don't use the `==` operator. Instead, compare them item-by-item in a loop or using iteration methods explained in the next chapter. + ## Summary Array is a special kind of object, suited to storing and managing ordered data items. -- The declaration: +The declaration: - ```js - // square brackets (usual) - let arr = [item1, item2...]; +```js +// square brackets (usual) +let arr = [item1, item2...]; - // new Array (exceptionally rare) - let arr = new Array(item1, item2...); - ``` +// new Array (exceptionally rare) +let arr = new Array(item1, item2...); +``` - The call to `new Array(number)` creates an array with the given length, but without elements. +The call to `new Array(number)` creates an array with the given length, but without elements. -- The `length` property is the array length or, to be precise, its last numeric index plus one. It is auto-adjusted by array methods. +- The `length` property is the array length or, to be precise, its last numeric index plus one. It is auto-adjusted by array methods. - If we shorten `length` manually, the array is truncated. +Getting the elements: + +- we can get element by its index, like `arr[0]` +- also we can use `at(i)` method that allows negative indexes. For negative values of `i`, it steps back from the end of the array. If `i >= 0`, it works same as `arr[i]`. + We can use an array as a deque with the following operations: - `push(...items)` adds `items` to the end. - `pop()` removes the element from the end and returns it. - `shift()` removes the element from the beginning and returns it. -- `unshift(...items)` adds items to the beginning. +- `unshift(...items)` adds `items` to the beginning. To loop over the elements of the array: - `for (let i=0; i<arr.length; i++)` -- works fastest, old-browser-compatible. - `for (let item of arr)` -- the modern syntax for items only, - `for (let i in arr)` -- never use. -We will return to arrays and study more methods to add, remove, extract elements and sort arrays in the chapter <info:array-methods>. +To compare arrays, don't use the `==` operator (as well as `>`, `<` and others), as they have no special treatment for arrays. They handle them as any objects, and it's not what we usually want. + +Instead you can use `for..of` loop to compare arrays item-by-item. +We will continue with arrays and study more methods to add, remove, extract elements and sort arrays in the next chapter <info:array-methods>. diff --git a/1-js/05-data-types/05-array-methods/10-average-age/task.md b/1-js/05-data-types/05-array-methods/10-average-age/task.md index a991c156b..bf5f85df3 100644 --- a/1-js/05-data-types/05-array-methods/10-average-age/task.md +++ b/1-js/05-data-types/05-array-methods/10-average-age/task.md @@ -4,7 +4,7 @@ importance: 4 # Get average age -Write the function `getAverageAge(users)` that gets an array of objects with property `age` and gets the average. +Write the function `getAverageAge(users)` that gets an array of objects with property `age` and returns the average age. The formula for the average is `(age1 + age2 + ... + ageN) / N`. @@ -19,4 +19,3 @@ let arr = [ john, pete, mary ]; alert( getAverageAge(arr) ); // (25 + 30 + 29) / 3 = 28 ``` - diff --git a/1-js/05-data-types/05-array-methods/11-array-unique/solution.md b/1-js/05-data-types/05-array-methods/11-array-unique/solution.md index 32d3b2679..b9d627a0a 100644 --- a/1-js/05-data-types/05-array-methods/11-array-unique/solution.md +++ b/1-js/05-data-types/05-array-methods/11-array-unique/solution.md @@ -36,4 +36,4 @@ So if `arr.length` is `10000` we'll have something like `10000*10000` = 100 mill So the solution is only good for small arrays. -Further in the chapter <info:map-set-weakmap-weakset> we'll see how to optimize it. +Further in the chapter <info:map-set> we'll see how to optimize it. diff --git a/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/solution.js b/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/solution.js new file mode 100644 index 000000000..8dea23a06 --- /dev/null +++ b/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/solution.js @@ -0,0 +1,6 @@ +function groupById(array) { + return array.reduce((obj, value) => { + obj[value.id] = value; + return obj; + }, {}) +} diff --git a/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/test.js b/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/test.js new file mode 100644 index 000000000..e48ba138d --- /dev/null +++ b/1-js/05-data-types/05-array-methods/12-reduce-object/_js.view/test.js @@ -0,0 +1,21 @@ +describe("groupById", function() { + + it("creates an object grouped by id", function() { + let users = [ + {id: 'john', name: "John Smith", age: 20}, + {id: 'ann', name: "Ann Smith", age: 24}, + {id: 'pete', name: "Pete Peterson", age: 31}, + ]; + + assert.deepEqual(groupById(users), { + john: {id: 'john', name: "John Smith", age: 20}, + ann: {id: 'ann', name: "Ann Smith", age: 24}, + pete: {id: 'pete', name: "Pete Peterson", age: 31}, + }); + }); + + it("works with an empty array", function() { + users = []; + assert.deepEqual(groupById(users), {}); + }); +}); diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/solution.md b/1-js/05-data-types/05-array-methods/12-reduce-object/solution.md similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/solution.md rename to 1-js/05-data-types/05-array-methods/12-reduce-object/solution.md diff --git a/1-js/05-data-types/05-array-methods/12-reduce-object/task.md b/1-js/05-data-types/05-array-methods/12-reduce-object/task.md new file mode 100644 index 000000000..7f0082357 --- /dev/null +++ b/1-js/05-data-types/05-array-methods/12-reduce-object/task.md @@ -0,0 +1,37 @@ +importance: 4 + +--- + +# Create keyed object from array + +Let's say we received an array of users in the form `{id:..., name:..., age:... }`. + +Create a function `groupById(arr)` that creates an object from it, with `id` as the key, and array items as values. + +For example: + +```js +let users = [ + {id: 'john', name: "John Smith", age: 20}, + {id: 'ann', name: "Ann Smith", age: 24}, + {id: 'pete', name: "Pete Peterson", age: 31}, +]; + +let usersById = groupById(users); + +/* +// after the call we should have: + +usersById = { + john: {id: 'john', name: "John Smith", age: 20}, + ann: {id: 'ann', name: "Ann Smith", age: 24}, + pete: {id: 'pete', name: "Pete Peterson", age: 31}, +} +*/ +``` + +Such function is really handy when working with server data. + +In this task we assume that `id` is unique. There may be no two array items with the same `id`. + +Please use array `.reduce` method in the solution. diff --git a/1-js/05-data-types/05-array-methods/2-filter-range/task.md b/1-js/05-data-types/05-array-methods/2-filter-range/task.md index 18b2c1d9b..46e47c93d 100644 --- a/1-js/05-data-types/05-array-methods/2-filter-range/task.md +++ b/1-js/05-data-types/05-array-methods/2-filter-range/task.md @@ -4,7 +4,7 @@ importance: 4 # Filter range -Write a function `filterRange(arr, a, b)` that gets an array `arr`, looks for elements between `a` and `b` in it and returns an array of them. +Write a function `filterRange(arr, a, b)` that gets an array `arr`, looks for elements with values higher or equal to `a` and lower or equal to `b` and return a result as an array. The function should not modify the array. It should return the new array. diff --git a/1-js/05-data-types/05-array-methods/3-filter-range-in-place/_js.view/test.js b/1-js/05-data-types/05-array-methods/3-filter-range-in-place/_js.view/test.js index db32d9a11..241b74c6e 100644 --- a/1-js/05-data-types/05-array-methods/3-filter-range-in-place/_js.view/test.js +++ b/1-js/05-data-types/05-array-methods/3-filter-range-in-place/_js.view/test.js @@ -4,13 +4,13 @@ describe("filterRangeInPlace", function() { let arr = [5, 3, 8, 1]; - filterRangeInPlace(arr, 1, 4); + filterRangeInPlace(arr, 2, 5); - assert.deepEqual(arr, [3, 1]); + assert.deepEqual(arr, [5, 3]); }); it("doesn't return anything", function() { assert.isUndefined(filterRangeInPlace([1,2,3], 1, 4)); }); -}); \ No newline at end of file +}); diff --git a/1-js/05-data-types/05-array-methods/4-sort-back/task.md b/1-js/05-data-types/05-array-methods/4-sort-back/task.md index 05a08aad0..0e3eeab76 100644 --- a/1-js/05-data-types/05-array-methods/4-sort-back/task.md +++ b/1-js/05-data-types/05-array-methods/4-sort-back/task.md @@ -2,12 +2,12 @@ importance: 4 --- -# Sort in the reverse order +# Sort in decreasing order ```js let arr = [5, 2, 1, -10, 8]; -// ... your code to sort it in the reverse order +// ... your code to sort it in decreasing order alert( arr ); // 8, 5, 2, 1, -10 ``` diff --git a/1-js/05-data-types/05-array-methods/6-calculator-extendable/_js.view/solution.js b/1-js/05-data-types/05-array-methods/6-calculator-extendable/_js.view/solution.js index 50c40e804..f62452a5f 100644 --- a/1-js/05-data-types/05-array-methods/6-calculator-extendable/_js.view/solution.js +++ b/1-js/05-data-types/05-array-methods/6-calculator-extendable/_js.view/solution.js @@ -1,6 +1,6 @@ function Calculator() { - let methods = { + this.methods = { "-": (a, b) => a - b, "+": (a, b) => a + b }; @@ -10,16 +10,16 @@ function Calculator() { let split = str.split(' '), a = +split[0], op = split[1], - b = +split[2] + b = +split[2]; - if (!methods[op] || isNaN(a) || isNaN(b)) { + if (!this.methods[op] || isNaN(a) || isNaN(b)) { return NaN; } - return methods[op](a, b); - } + return this.methods[op](a, b); + }; this.addMethod = function(name, func) { - methods[name] = func; + this.methods[name] = func; }; } diff --git a/1-js/05-data-types/05-array-methods/6-calculator-extendable/solution.md b/1-js/05-data-types/05-array-methods/6-calculator-extendable/solution.md index 41178663d..ebe0714cf 100644 --- a/1-js/05-data-types/05-array-methods/6-calculator-extendable/solution.md +++ b/1-js/05-data-types/05-array-methods/6-calculator-extendable/solution.md @@ -1,3 +1,3 @@ -- Please note how methods are stored. They are simply added to the internal object. +- Please note how methods are stored. They are simply added to `this.methods` property. - All tests and numeric conversions are done in the `calculate` method. In future it may be extended to support more complex expressions. diff --git a/1-js/05-data-types/05-array-methods/6-calculator-extendable/task.md b/1-js/05-data-types/05-array-methods/6-calculator-extendable/task.md index cc5453ceb..e0d302f4c 100644 --- a/1-js/05-data-types/05-array-methods/6-calculator-extendable/task.md +++ b/1-js/05-data-types/05-array-methods/6-calculator-extendable/task.md @@ -31,6 +31,6 @@ The task consists of two parts. alert( result ); // 8 ``` -- No brackets or complex expressions in this task. +- No parentheses or complex expressions in this task. - The numbers and the operator are delimited with exactly one space. - There may be error handling if you'd like to add it. diff --git a/1-js/05-data-types/05-array-methods/7-map-objects/solution.md b/1-js/05-data-types/05-array-methods/7-map-objects/solution.md index 5d8bf4a13..2d8d4fb0e 100644 --- a/1-js/05-data-types/05-array-methods/7-map-objects/solution.md +++ b/1-js/05-data-types/05-array-methods/7-map-objects/solution.md @@ -25,7 +25,7 @@ alert( usersMapped[0].id ); // 1 alert( usersMapped[0].fullName ); // John Smith ``` -Please note that in for the arrow functions we need to use additional brackets. +Please note that in the arrow functions we need to use additional brackets. We can't write like this: ```js diff --git a/1-js/05-data-types/05-array-methods/8-sort-objects/solution.md b/1-js/05-data-types/05-array-methods/8-sort-objects/solution.md index 9f1ade707..cfaf9761a 100644 --- a/1-js/05-data-types/05-array-methods/8-sort-objects/solution.md +++ b/1-js/05-data-types/05-array-methods/8-sort-objects/solution.md @@ -1,6 +1,6 @@ ```js run no-beautify function sortByAge(arr) { - arr.sort((a, b) => a.age > b.age ? 1 : -1); + arr.sort((a, b) => a.age - b.age); } let john = { name: "John", age: 25 }; diff --git a/1-js/05-data-types/05-array-methods/9-shuffle/solution.md b/1-js/05-data-types/05-array-methods/9-shuffle/solution.md index a43715db8..6674c444f 100644 --- a/1-js/05-data-types/05-array-methods/9-shuffle/solution.md +++ b/1-js/05-data-types/05-array-methods/9-shuffle/solution.md @@ -45,7 +45,7 @@ for (let key in count) { } ``` -An example result (for V8, July 2017): +An example result (depends on JS engine): ```js 123: 250706 @@ -68,7 +68,13 @@ There are other good ways to do the task. For instance, there's a great algorith function shuffle(array) { for (let i = array.length - 1; i > 0; i--) { let j = Math.floor(Math.random() * (i + 1)); // random index from 0 to i - [array[i], array[j]] = [array[j], array[i]]; // swap elements + + // swap elements array[i] and array[j] + // we use "destructuring assignment" syntax to achieve that + // you'll find more details about that syntax in later chapters + // same can be written as: + // let t = array[i]; array[i] = array[j]; array[j] = t + [array[i], array[j]] = [array[j], array[i]]; } } ``` diff --git a/1-js/05-data-types/05-array-methods/article.md b/1-js/05-data-types/05-array-methods/article.md index 049fd6e56..853645958 100644 --- a/1-js/05-data-types/05-array-methods/article.md +++ b/1-js/05-data-types/05-array-methods/article.md @@ -1,6 +1,6 @@ # Array methods -Arrays provide a lot of methods. To make things easier, in this chapter they are split into groups. +Arrays provide a lot of methods. To make things easier, in this chapter, they are split into groups. ## Add/remove items @@ -11,7 +11,7 @@ We already know methods that add and remove items from the beginning or the end: - `arr.shift()` -- extracts an item from the beginning, - `arr.unshift(...items)` -- adds items to the beginning. -Here are few others. +Here are a few others. ### splice @@ -32,19 +32,19 @@ alert( arr.length ); // 3 The element was removed, but the array still has 3 elements, we can see that `arr.length == 3`. -That's natural, because `delete obj.key` removes a value by the `key`. It's all it does. Fine for objects. But for arrays we usually want the rest of elements to shift and occupy the freed place. We expect to have a shorter array now. +That's natural, because `delete obj.key` removes a value by the `key`. It's all it does. Fine for objects. But for arrays we usually want the rest of the elements to shift and occupy the freed place. We expect to have a shorter array now. So, special methods should be used. -The [arr.splice(str)](mdn:js/Array/splice) method is a swiss army knife for arrays. It can do everything: insert, remove and replace elements. +The [arr.splice](mdn:js/Array/splice) method is a Swiss army knife for arrays. It can do everything: insert, remove and replace elements. The syntax is: ```js -arr.splice(index[, deleteCount, elem1, ..., elemN]) +arr.splice(start[, deleteCount, elem1, ..., elemN]) ``` -It starts from the position `index`: removes `deleteCount` elements and then inserts `elem1, ..., elemN` at their place. Returns the array of removed elements. +It modifies `arr` starting from the index `start`: removes `deleteCount` elements and then inserts `elem1, ..., elemN` at their place. Returns the array of removed elements. This method is easy to grasp by examples. @@ -62,7 +62,7 @@ alert( arr ); // ["I", "JavaScript"] Easy, right? Starting from the index `1` it removed `1` element. -In the next example we remove 3 elements and replace them with the other two: +In the next example, we remove 3 elements and replace them with the other two: ```js run let arr = [*!*"I", "study", "JavaScript",*/!* "right", "now"]; @@ -84,7 +84,7 @@ let removed = arr.splice(0, 2); alert( removed ); // "I", "study" <-- array of removed elements ``` -The `splice` method is also able to insert the elements without any removals. For that we need to set `deleteCount` to `0`: +The `splice` method is also able to insert the elements without any removals. For that, we need to set `deleteCount` to `0`: ```js run let arr = ["I", "study", "JavaScript"]; @@ -114,34 +114,33 @@ alert( arr ); // 1,2,3,4,5 ### slice -The method [arr.slice](mdn:js/Array/slice) is much simpler than similar-looking `arr.splice`. +The method [arr.slice](mdn:js/Array/slice) is much simpler than the similar-looking `arr.splice`. The syntax is: ```js -arr.slice(start, end) +arr.slice([start], [end]) ``` -It returns a new array containing all items from index `"start"` to `"end"` (not including `"end"`). Both `start` and `end` can be negative, in that case position from array end is assumed. +It returns a new array copying to it all items from index `start` to `end` (not including `end`). Both `start` and `end` can be negative, in that case position from array end is assumed. -It works like `str.slice`, but makes subarrays instead of substrings. +It's similar to a string method `str.slice`, but instead of substrings, it makes subarrays. For instance: ```js run -let str = "test"; let arr = ["t", "e", "s", "t"]; -alert( str.slice(1, 3) ); // es -alert( arr.slice(1, 3) ); // e,s +alert( arr.slice(1, 3) ); // e,s (copy from 1 to 3) -alert( str.slice(-2) ); // st -alert( arr.slice(-2) ); // s,t +alert( arr.slice(-2) ); // s,t (copy from -2 till the end) ``` +We can also call it without arguments: `arr.slice()` creates a copy of `arr`. That's often used to obtain a copy for further transformations that should not affect the original array. + ### concat -The method [arr.concat](mdn:js/Array/concat) joins the array with other arrays and/or items. +The method [arr.concat](mdn:js/Array/concat) creates a new array that includes values from other arrays and additional items. The syntax is: @@ -153,24 +152,24 @@ It accepts any number of arguments -- either arrays or values. The result is a new array containing items from `arr`, then `arg1`, `arg2` etc. -If an argument is an array or has `Symbol.isConcatSpreadable` property, then all its elements are copied. Otherwise, the argument itself is copied. +If an argument `argN` is an array, then all its elements are copied. Otherwise, the argument itself is copied. For instance: ```js run let arr = [1, 2]; -// merge arr with [3,4] -alert( arr.concat([3, 4])); // 1,2,3,4 +// create an array from: arr and [3,4] +alert( arr.concat([3, 4]) ); // 1,2,3,4 -// merge arr with [3,4] and [5,6] -alert( arr.concat([3, 4], [5, 6])); // 1,2,3,4,5,6 +// create an array from: arr and [3,4] and [5,6] +alert( arr.concat([3, 4], [5, 6]) ); // 1,2,3,4,5,6 -// merge arr with [3,4], then add values 5 and 6 -alert( arr.concat([3, 4], 5, 6)); // 1,2,3,4,5,6 +// create an array from: arr and [3,4], then add values 5 and 6 +alert( arr.concat([3, 4], 5, 6) ); // 1,2,3,4,5,6 ``` -Normally, it only copies elements from arrays ("spreads" them). Other objects, even if they look like arrays, added as a whole: +Normally, it only copies elements from arrays. Other objects, even if they look like arrays, are added as a whole: ```js run let arr = [1, 2]; @@ -181,10 +180,9 @@ let arrayLike = { }; alert( arr.concat(arrayLike) ); // 1,2,[object Object] -//[1, 2, arrayLike] ``` -...But if an array-like object has `Symbol.isConcatSpreadable` property, then its elements are added instead: +...But if an array-like object has a special `Symbol.isConcatSpreadable` property, then it's treated as an array by `concat`: its elements are added instead: ```js run let arr = [1, 2]; @@ -208,7 +206,7 @@ The [arr.forEach](mdn:js/Array/forEach) method allows to run a function for ever The syntax: ```js arr.forEach(function(item, index, array) { - // ... do something with item + // ... do something with an item }); ``` @@ -232,16 +230,17 @@ The result of the function (if it returns any) is thrown away and ignored. ## Searching in array -These are methods to search for something in an array. +Now let's cover methods that search in an array. ### indexOf/lastIndexOf and includes -The methods [arr.indexOf](mdn:js/Array/indexOf), [arr.lastIndexOf](mdn:js/Array/lastIndexOf) and [arr.includes](mdn:js/Array/includes) have the same syntax and do essentially the same as their string counterparts, but operate on items instead of characters: +The methods [arr.indexOf](mdn:js/Array/indexOf) and [arr.includes](mdn:js/Array/includes) have the similar syntax and do essentially the same as their string counterparts, but operate on items instead of characters: -- `arr.indexOf(item, from)` looks for `item` starting from index `from`, and returns the index where it was found, otherwise `-1`. -- `arr.lastIndexOf(item, from)` -- same, but looks for from right to left. +- `arr.indexOf(item, from)` -- looks for `item` starting from index `from`, and returns the index where it was found, otherwise `-1`. - `arr.includes(item, from)` -- looks for `item` starting from index `from`, returns `true` if found. +Usually, these methods are used with only one argument: the `item` to search. By default, the search is from the beginning. + For instance: ```js run @@ -254,23 +253,35 @@ alert( arr.indexOf(null) ); // -1 alert( arr.includes(1) ); // true ``` -Note that the methods use `===` comparison. So, if we look for `false`, it finds exactly `false` and not the zero. +Please note that `indexOf` uses the strict equality `===` for comparison. So, if we look for `false`, it finds exactly `false` and not the zero. + +If we want to check if `item` exists in the array and don't need the index, then `arr.includes` is preferred. + +The method [arr.lastIndexOf](mdn:js/Array/lastIndexOf) is the same as `indexOf`, but looks for from right to left. + +```js run +let fruits = ['Apple', 'Orange', 'Apple'] -If we want to check for inclusion, and don't want to know the exact index, then `arr.includes` is preferred. +alert( fruits.indexOf('Apple') ); // 0 (first Apple) +alert( fruits.lastIndexOf('Apple') ); // 2 (last Apple) +``` -Also, a very minor difference of `includes` is that it correctly handles `NaN`, unlike `indexOf/lastIndexOf`: +````smart header="The `includes` method handles `NaN` correctly" +A minor, but noteworthy feature of `includes` is that it correctly handles `NaN`, unlike `indexOf`: ```js run const arr = [NaN]; -alert( arr.indexOf(NaN) ); // -1 (should be 0, but === equality doesn't work for NaN) +alert( arr.indexOf(NaN) ); // -1 (wrong, should be 0) alert( arr.includes(NaN) );// true (correct) ``` +That's because `includes` was added to JavaScript much later and uses the more up-to-date comparison algorithm internally. +```` -### find and findIndex +### find and findIndex/findLastIndex -Imagine we have an array of objects. How do we find an object with the specific condition? +Imagine we have an array of objects. How do we find an object with a specific condition? -Here the [arr.find](mdn:js/Array/find) method comes in handy. +Here the [arr.find(fn)](mdn:js/Array/find) method comes in handy. The syntax is: ```js @@ -280,13 +291,13 @@ let result = arr.find(function(item, index, array) { }); ``` -The function is called repetitively for each element of the array: +The function is called for elements of the array, one after another: - `item` is the element. - `index` is its index. - `array` is the array itself. -If it returns `true`, the search is stopped, the `item` is returned. If nothing found, `undefined` is returned. +If it returns `true`, the search is stopped, the `item` is returned. If nothing is found, `undefined` is returned. For example, we have an array of users, each with the fields `id` and `name`. Let's find the one with `id == 1`: @@ -302,11 +313,30 @@ let user = users.find(item => item.id == 1); alert(user.name); // John ``` -In real life arrays of objects is a common thing, so the `find` method is very useful. +In real life, arrays of objects are a common thing, so the `find` method is very useful. + +Note that in the example we provide to `find` the function `item => item.id == 1` with one argument. That's typical, other arguments of this function are rarely used. + +The [arr.findIndex](mdn:js/Array/findIndex) method has the same syntax but returns the index where the element was found instead of the element itself. The value of `-1` is returned if nothing is found. + +The [arr.findLastIndex](mdn:js/Array/findLastIndex) method is like `findIndex`, but searches from right to left, similar to `lastIndexOf`. + +Here's an example: + +```js run +let users = [ + {id: 1, name: "John"}, + {id: 2, name: "Pete"}, + {id: 3, name: "Mary"}, + {id: 4, name: "John"} +]; -Note that in the example we provide to `find` the function `item => item.id == 1` with one argument. Other arguments of this function are rarely used. +// Find the index of the first John +alert(users.findIndex(user => user.name == 'John')); // 0 -The [arr.findIndex](mdn:js/Array/findIndex) method is essentially the same, but it returns the index where the element was found instead of the element itself and `-1` is returned when nothing is found. +// Find the index of the last John +alert(users.findLastIndex(user => user.name == 'John')); // 3 +``` ### filter @@ -314,12 +344,12 @@ The `find` method looks for a single (first) element that makes the function ret If there may be many, we can use [arr.filter(fn)](mdn:js/Array/filter). -The syntax is similar to `find`, but filter continues to iterate for all array elements even if `true` is already returned: +The syntax is similar to `find`, but `filter` returns an array of all matching elements: ```js let results = arr.filter(function(item, index, array) { - // if true item is pushed to results and iteration continues - // returns empty array for complete falsy scenario + // if true item is pushed to results and the iteration continues + // returns empty array if nothing found }); ``` @@ -340,23 +370,22 @@ alert(someUsers.length); // 2 ## Transform an array -This section is about the methods transforming or reordering the array. - +Let's move on to methods that transform and reorder an array. ### map The [arr.map](mdn:js/Array/map) method is one of the most useful and often used. +It calls the function for each element of the array and returns the array of results. + The syntax is: ```js let result = arr.map(function(item, index, array) { // returns the new value instead of item -}) +}); ``` -It calls the function for each element of the array and returns the array of results. - For instance, here we transform each element into its length: ```js run @@ -366,14 +395,16 @@ alert(lengths); // 5,7,6 ### sort(fn) -The method [arr.sort](mdn:js/Array/sort) sorts the array *in place*. +The call to [arr.sort()](mdn:js/Array/sort) sorts the array *in place*, changing its element order. + +It also returns the sorted array, but the returned value is usually ignored, as `arr` itself is modified. For instance: ```js run let arr = [ 1, 2, 15 ]; -// the method reorders the content of arr (and returns it) +// the method reorders the content of arr arr.sort(); alert( arr ); // *!*1, 15, 2*/!* @@ -385,20 +416,21 @@ The order became `1, 15, 2`. Incorrect. But why? **The items are sorted as strings by default.** -Literally, all elements are converted to strings and then compared. So, the lexicographic ordering is applied and indeed `"2" > "15"`. +Literally, all elements are converted to strings for comparisons. For strings, lexicographic ordering is applied and indeed `"2" > "15"`. -To use our own sorting order, we need to supply a function of two arguments as the argument of `arr.sort()`. +To use our own sorting order, we need to supply a function as the argument of `arr.sort()`. + +The function should compare two arbitrary values and return: -The function should work like this: ```js function compare(a, b) { - if (a > b) return 1; - if (a == b) return 0; - if (a < b) return -1; + if (a > b) return 1; // if the first value is greater than the second + if (a == b) return 0; // if values are equal + if (a < b) return -1; // if the first value is less than the second } ``` -For instance: +For instance, to sort as numbers: ```js run function compareNumeric(a, b) { @@ -418,20 +450,20 @@ alert(arr); // *!*1, 2, 15*/!* Now it works as intended. -Let's step aside and think what's happening. The `arr` can be array of anything, right? It may contain numbers or strings or html elements or whatever. We have a set of *something*. To sort it, we need an *ordering function* that knows how to compare its elements. The default is a string order. +Let's step aside and think about what's happening. The `arr` can be an array of anything, right? It may contain numbers or strings or objects or whatever. We have a set of *some items*. To sort it, we need an *ordering function* that knows how to compare its elements. The default is a string order. -The `arr.sort(fn)` method has a built-in implementation of sorting algorithm. We don't need to care how it exactly works (an optimized [quicksort](https://en.wikipedia.org/wiki/Quicksort) most of the time). It will walk the array, compare its elements using the provided function and reorder them, all we need is to provide the `fn` which does the comparison. +The `arr.sort(fn)` method implements a generic sorting algorithm. We don't need to care how it internally works (an optimized [quicksort](https://en.wikipedia.org/wiki/Quicksort) or [Timsort](https://en.wikipedia.org/wiki/Timsort) most of the time). It will walk the array, compare its elements using the provided function and reorder them, all we need is to provide the `fn` which does the comparison. -By the way, if we ever want to know which elements are compared -- nothing prevents from alerting them: +By the way, if we ever want to know which elements are compared -- nothing prevents us from alerting them: ```js run [1, -2, 15, 2, 0, 8].sort(function(a, b) { alert( a + " <> " + b ); + return a - b; }); ``` -The algorithm may compare an element multiple times in the process, but it tries to make as few comparisons as possible. - +The algorithm may compare an element with multiple others in the process, but it tries to make as few comparisons as possible. ````smart header="A comparison function may return any number" Actually, a comparison function is only required to return a positive number to say "greater" and a negative number to say "less". @@ -448,13 +480,29 @@ alert(arr); // *!*1, 2, 15*/!* ```` ````smart header="Arrow functions for the best" -Remember [arrow functions](info:function-expressions-arrows#arrow-functions)? We can use them here for neater sorting: +Remember [arrow functions](info:arrow-functions-basics)? We can use them here for neater sorting: ```js arr.sort( (a, b) => a - b ); ``` -This works exactly the same as the other, longer, version above. +This works exactly the same as the longer version above. +```` + +````smart header="Use `localeCompare` for strings" +Remember [strings](info:string#correct-comparisons) comparison algorithm? It compares letters by their codes by default. + +For many alphabets, it's better to use `str.localeCompare` method to correctly sort letters, such as `Ö`. + +For example, let's sort a few countries in German: + +```js run +let countries = ['Österreich', 'Andorra', 'Vietnam']; + +alert( countries.sort( (a, b) => a > b ? 1 : -1) ); // Andorra, Vietnam, Österreich (wrong) + +alert( countries.sort( (a, b) => a.localeCompare(b) ) ); // Andorra,Österreich,Vietnam (correct!) +``` ```` ### reverse @@ -474,11 +522,11 @@ It also returns the array `arr` after the reversal. ### split and join -Here's the situation from the real life. We are writing a messaging app, and the person enters the comma-delimited list of receivers: `John, Pete, Mary`. But for us an array of names would be much more comfortable than a single string. How to get it? +Here's the situation from real life. We are writing a messaging app, and the person enters the comma-delimited list of receivers: `John, Pete, Mary`. But for us an array of names would be much more comfortable than a single string. How to get it? The [str.split(delim)](mdn:js/String/split) method does exactly that. It splits the string into an array by the given delimiter `delim`. -In the example below, we split by a comma followed by space: +In the example below, we split by a comma followed by a space: ```js run let names = 'Bilbo, Gandalf, Nazgul'; @@ -508,14 +556,14 @@ alert( str.split('') ); // t,e,s,t ``` ```` -The call [arr.join(separator)](mdn:js/Array/join) does the reverse to `split`. It creates a string of `arr` items glued by `separator` between them. +The call [arr.join(glue)](mdn:js/Array/join) does the reverse to `split`. It creates a string of `arr` items joined by `glue` between them. For instance: ```js run let arr = ['Bilbo', 'Gandalf', 'Nazgul']; -let str = arr.join(';'); +let str = arr.join(';'); // glue the array into a string using ; alert( str ); // Bilbo;Gandalf;Nazgul ``` @@ -531,24 +579,29 @@ The methods [arr.reduce](mdn:js/Array/reduce) and [arr.reduceRight](mdn:js/Array The syntax is: ```js -let value = arr.reduce(function(previousValue, item, index, array) { +let value = arr.reduce(function(accumulator, item, index, array) { // ... -}, initial); +}, [initial]); ``` -The function is applied to the elements. You may notice the familiar arguments, starting from the 2nd: +The function is applied to all array elements one after another and "carries on" its result to the next call. + +Arguments: +- `accumulator` -- is the result of the previous function call, equals `initial` the first time (if `initial` is provided). - `item` -- is the current array item. - `index` -- is its position. - `array` -- is the array. -So far, like `forEach/map`. But there's one more argument: +As the function is applied, the result of the previous function call is passed to the next one as the first argument. -- `previousValue` -- is the result of the previous function call, `initial` for the first call. +So, the first argument is essentially the accumulator that stores the combined result of all previous executions. And at the end, it becomes the result of `reduce`. + +Sounds complicated? The easiest way to grasp that is by example. -Here we get a sum of array in one line: +Here we get a sum of an array in one line: ```js run let arr = [1, 2, 3, 4, 5]; @@ -558,11 +611,11 @@ let result = arr.reduce((sum, current) => sum + current, 0); alert(result); // 15 ``` -Here we used the most common variant of `reduce` which uses only 2 arguments. +The function passed to `reduce` uses only 2 arguments, that's typically enough. Let's see the details of what's going on. -1. On the first run, `sum` is the initial value (the last argument of `reduce`), equals `0`, and `current` is the first array element, equals `1`. So the result is `1`. +1. On the first run, `sum` is the `initial` value (the last argument of `reduce`), equals `0`, and `current` is the first array element, equals `1`. So the function result is `1`. 2. On the second run, `sum = 1`, we add the second array element (`2`) to it and return. 3. On the 3rd run, `sum = 3` and we add one more element to it, and so on... @@ -572,7 +625,7 @@ The calculation flow: Or in the form of a table, where each row represents a function call on the next array element: -| |`sum`|`current`|`result`| +| |`sum`|`current`|result| |---|-----|---------|---------| |the first call|`0`|`1`|`1`| |the second call|`1`|`2`|`3`| @@ -580,8 +633,7 @@ Or in the form of a table, where each row represents a function call on the next |the fourth call|`6`|`4`|`10`| |the fifth call|`10`|`5`|`15`| - -As we can see, the result of the previous call becomes the first argument of the next one. +Here we can clearly see how the result of the previous call becomes the first argument of the next one. We also can omit the initial value: @@ -610,11 +662,9 @@ let arr = []; arr.reduce((sum, current) => sum + current); ``` - So it's advised to always specify the initial value. -The method [arr.reduceRight](mdn:js/Array/reduceRight) does the same, but goes from right to left. - +The method [arr.reduceRight](mdn:js/Array/reduceRight) does the same but goes from right to left. ## Array.isArray @@ -624,7 +674,7 @@ So `typeof` does not help to distinguish a plain object from an array: ```js run alert(typeof {}); // object -alert(typeof []); // same +alert(typeof []); // object (same) ``` ...But arrays are used so often that there's a special method for that: [Array.isArray(value)](mdn:js/Array/isArray). It returns `true` if the `value` is an array, and `false` otherwise. @@ -639,7 +689,7 @@ alert(Array.isArray([])); // true Almost all array methods that call functions -- like `find`, `filter`, `map`, with a notable exception of `sort`, accept an optional additional parameter `thisArg`. -That parameter is not explained in the sections above, because it's rarely used. But for completeness we have to cover it. +That parameter is not explained in the sections above, because it's rarely used. But for completeness, we have to cover it. Here's the full syntax of these methods: @@ -653,51 +703,57 @@ arr.map(func, thisArg); The value of `thisArg` parameter becomes `this` for `func`. -For instance, here we use an object method as a filter and `thisArg` comes in handy: +For example, here we use a method of `army` object as a filter, and `thisArg` passes the context: ```js run -let user = { - age: 18, - younger(otherUser) { - return otherUser.age < this.age; +let army = { + minAge: 18, + maxAge: 27, + canJoin(user) { + return user.age >= this.minAge && user.age < this.maxAge; } }; let users = [ - {age: 12}, {age: 16}, - {age: 32} + {age: 20}, + {age: 23}, + {age: 30} ]; *!* -// find all users younger than user -let youngerUsers = users.filter(user.younger, user); +// find users, for who army.canJoin returns true +let soldiers = users.filter(army.canJoin, army); */!* -alert(youngerUsers.length); // 2 +alert(soldiers.length); // 2 +alert(soldiers[0].age); // 20 +alert(soldiers[1].age); // 23 ``` -In the call above, we use `user.younger` as a filter and also provide `user` as the context for it. If we didn't provide the context, `users.filter(user.younger)` would call `user.younger` as a standalone function, with `this=undefined`. That would mean an instant error. +If in the example above we used `users.filter(army.canJoin)`, then `army.canJoin` would be called as a standalone function, with `this=undefined`, thus leading to an instant error. + +A call to `users.filter(army.canJoin, army)` can be replaced with `users.filter(user => army.canJoin(user))`, that does the same. The latter is used more often, as it's a bit easier to understand for most people. ## Summary -A cheatsheet of array methods: +A cheat sheet of array methods: - To add/remove elements: - `push(...items)` -- adds items to the end, - `pop()` -- extracts an item from the end, - `shift()` -- extracts an item from the beginning, - `unshift(...items)` -- adds items to the beginning. - - `splice(pos, deleteCount, ...items)` -- at index `pos` delete `deleteCount` elements and insert `items`. - - `slice(start, end)` -- creates a new array, copies elements from position `start` till `end` (not inclusive) into it. + - `splice(pos, deleteCount, ...items)` -- at index `pos` deletes `deleteCount` elements and inserts `items`. + - `slice(start, end)` -- creates a new array, copies elements from index `start` till `end` (not inclusive) into it. - `concat(...items)` -- returns a new array: copies all members of the current one and adds `items` to it. If any of `items` is an array, then its elements are taken. - To search among elements: - - `indexOf/lastIndexOf(item, pos)` -- look for `item` starting from position `pos`, return the index or `-1` if not found. + - `indexOf/lastIndexOf(item, pos)` -- look for `item` starting from position `pos`, and return the index or `-1` if not found. - `includes(value)` -- returns `true` if the array has `value`, otherwise `false`. - `find/filter(func)` -- filter elements through the function, return first/all values that make it return `true`. - `findIndex` is like `find`, but returns the index instead of a value. - + - To iterate over elements: - `forEach(func)` -- calls `func` for every element, does not return anything. @@ -706,27 +762,41 @@ A cheatsheet of array methods: - `sort(func)` -- sorts the array in-place, then returns it. - `reverse()` -- reverses the array in-place, then returns it. - `split/join` -- convert a string to array and back. - - `reduce(func, initial)` -- calculate a single value over the array by calling `func` for each element and passing an intermediate result between the calls. + - `reduce/reduceRight(func, initial)` -- calculate a single value over the array by calling `func` for each element and passing an intermediate result between the calls. - Additionally: - - `Array.isArray(arr)` checks `arr` for being an array. + - `Array.isArray(value)` checks `value` for being an array, if so returns `true`, otherwise `false`. Please note that methods `sort`, `reverse` and `splice` modify the array itself. These methods are the most used ones, they cover 99% of use cases. But there are few others: -- [arr.some(fn)](mdn:js/Array/some)/[arr.every(fn)](mdn:js/Array/every) checks the array. +- [arr.some(fn)](mdn:js/Array/some)/[arr.every(fn)](mdn:js/Array/every) check the array. The function `fn` is called on each element of the array similar to `map`. If any/all results are `true`, returns `true`, otherwise `false`. + These methods behave sort of like `||` and `&&` operators: if `fn` returns a truthy value, `arr.some()` immediately returns `true` and stops iterating over the rest of items; if `fn` returns a falsy value, `arr.every()` immediately returns `false` and stops iterating over the rest of items as well. + + We can use `every` to compare arrays: + + ```js run + function arraysEqual(arr1, arr2) { + return arr1.length === arr2.length && arr1.every((value, index) => value === arr2[index]); + } + + alert( arraysEqual([1, 2], [1, 2])); // true + ``` + - [arr.fill(value, start, end)](mdn:js/Array/fill) -- fills the array with repeating `value` from index `start` to `end`. - [arr.copyWithin(target, start, end)](mdn:js/Array/copyWithin) -- copies its elements from position `start` till position `end` into *itself*, at position `target` (overwrites existing). +- [arr.flat(depth)](mdn:js/Array/flat)/[arr.flatMap(fn)](mdn:js/Array/flatMap) create a new flat array from a multidimensional array. + For the full list, see the [manual](mdn:js/Array). -From the first sight it may seem that there are so many methods, quite difficult to remember. But actually that's much easier than it seems. +At first sight, it may seem that there are so many methods, quite difficult to remember. But actually, that's much easier. -Look through the cheatsheet just to be aware of them. Then solve the tasks of this chapter to practice, so that you have experience with array methods. +Look through the cheat sheet just to be aware of them. Then solve the tasks of this chapter to practice, so that you have experience with array methods. -Afterwards whenever you need to do something with an array, and you don't know how -- come here, look at the cheatsheet and find the right method. Examples will help you to write it correctly. Soon you'll automatically remember the methods, without specific efforts from your side. +Afterwards whenever you need to do something with an array, and you don't know how -- come here, look at the cheat sheet and find the right method. Examples will help you to write it correctly. Soon you'll automatically remember the methods, without specific efforts from your side. diff --git a/1-js/05-data-types/06-iterable/article.md b/1-js/05-data-types/06-iterable/article.md index 7f72f5caf..e2c0d4f97 100644 --- a/1-js/05-data-types/06-iterable/article.md +++ b/1-js/05-data-types/06-iterable/article.md @@ -1,18 +1,18 @@ # Iterables -*Iterable* objects is a generalization of arrays. That's a concept that allows to make any object useable in a `for..of` loop. +*Iterable* objects are a generalization of arrays. That's a concept that allows us to make any object useable in a `for..of` loop. -Of course, Arrays are iterable. But there are many other built-in objects, that are iterable as well. For instance, Strings are iterable also. As we'll see, many built-in operators and methods rely on them. +Of course, Arrays are iterable. But there are many other built-in objects, that are iterable as well. For instance, strings are also iterable. -If an object represents a collection (list, set) of something, then `for..of` is a great syntax to loop over it, so let's see how to make it work. +If an object isn't technically an array, but represents a collection (list, set) of something, then `for..of` is a great syntax to loop over it, so let's see how to make it work. ## Symbol.iterator We can easily grasp the concept of iterables by making one of our own. -For instance, we have an object, that is not an array, but looks suitable for `for..of`. +For instance, we have an object that is not an array, but looks suitable for `for..of`. Like a `range` object that represents an interval of numbers: @@ -26,14 +26,14 @@ let range = { // for(let num of range) ... num=1,2,3,4,5 ``` -To make the `range` iterable (and thus let `for..of` work) we need to add a method to the object named `Symbol.iterator` (a special built-in symbol just for that). +To make the `range` object iterable (and thus let `for..of` work) we need to add a method to the object named `Symbol.iterator` (a special built-in symbol just for that). 1. When `for..of` starts, it calls that method once (or errors if not found). The method must return an *iterator* -- an object with the method `next`. 2. Onward, `for..of` works *only with that returned object*. 3. When `for..of` wants the next value, it calls `next()` on that object. -4. The result of `next()` must have the form `{done: Boolean, value: any}`, where `done=true` means that the iteration is finished, otherwise `value` must be the new value. +4. The result of `next()` must have the form `{done: Boolean, value: any}`, where `done=true` means that the loop is finished, otherwise `value` is the next value. -Here's the full implementation for `range`: +Here's the full implementation for `range` with remarks: ```js run let range = { @@ -45,10 +45,10 @@ let range = { range[Symbol.iterator] = function() { // ...it returns the iterator object: - // 2. Onward, for..of works only with this iterator, asking it for next values + // 2. Onward, for..of works only with the iterator object below, asking it for next values return { current: this.from, - last: this.to, + last: this.to, // 3. next() is called on each iteration by the for..of loop next() { @@ -68,10 +68,10 @@ for (let num of range) { } ``` -Please note the core feature of iterables: an important separation of concerns: +Please note the core feature of iterables: separation of concerns. - The `range` itself does not have the `next()` method. -- Instead, another object, a so-called "iterator" is created by the call to `range[Symbol.iterator]()`, and it handles the whole iteration. +- Instead, another object, a so-called "iterator" is created by the call to `range[Symbol.iterator]()`, and its `next()` generates values for the iteration. So, the iterator object is separate from the object it iterates over. @@ -105,7 +105,7 @@ for (let num of range) { Now `range[Symbol.iterator]()` returns the `range` object itself: it has the necessary `next()` method and remembers the current iteration progress in `this.current`. Shorter? Yes. And sometimes that's fine too. -The downside is that now it's impossible to have two `for..of` loops running over the object simultaneously: they'll share the iteration state, because there's only one iterator -- the object itself. But two parallel for-ofs is a rare thing, doable with some async scenarios. +The downside is that now it's impossible to have two `for..of` loops running over the object simultaneously: they'll share the iteration state, because there's only one iterator -- the object itself. But two parallel for-ofs is a rare thing, even in async scenarios. ```smart header="Infinite iterators" Infinite iterators are also possible. For instance, the `range` becomes infinite for `range.to = Infinity`. Or we can make an iterable object that generates an infinite sequence of pseudorandom numbers. Also can be useful. @@ -140,11 +140,9 @@ for (let char of str) { ## Calling an iterator explicitly -Normally, internals of iterables are hidden from the external code. There's a `for..of` loop, that works, that's all it needs to know. +For deeper understanding, let's see how to use an iterator explicitly. -But to understand things a little bit deeper let's see how to create an iterator explicitly. - -We'll iterate over a string the same way as `for..of`, but with direct calls. This code gets a string iterator and calls it "manually": +We'll iterate over a string in exactly the same way as `for..of`, but with direct calls. This code creates a string iterator and gets values from it "manually": ```js run let str = "Hello"; @@ -152,7 +150,9 @@ let str = "Hello"; // does the same as // for (let char of str) alert(char); +*!* let iterator = str[Symbol.iterator](); +*/!* while (true) { let result = iterator.next(); @@ -165,14 +165,16 @@ That is rarely needed, but gives us more control over the process than `for..of` ## Iterables and array-likes [#array-like] -There are two official terms that look similar, but are very different. Please make sure you understand them well to avoid the confusion. +Two official terms look similar, but are very different. Please make sure you understand them well to avoid the confusion. - *Iterables* are objects that implement the `Symbol.iterator` method, as described above. - *Array-likes* are objects that have indexes and `length`, so they look like arrays. -Naturally, these properties can combine. For instance, strings are both iterable (`for..of` works on them) and array-like (they have numeric indexes and `length`). +When we use JavaScript for practical tasks in a browser or any other environment, we may meet objects that are iterables or array-likes, or both. + +For instance, strings are both iterable (`for..of` works on them) and array-like (they have numeric indexes and `length`). -But an iterable may be not array-like. And vice versa an array-like may be not iterable. +But an iterable may not be array-like. And vice versa an array-like may not be iterable. For example, the `range` in the example above is iterable, but not array-like, because it does not have indexed properties and `length`. @@ -191,11 +193,11 @@ for (let item of arrayLike) {} */!* ``` -What do they have in common? Both iterables and array-likes are usually *not arrays*, they don't have `push`, `pop` etc. That's rather inconvenient if we have such an object and want to work with it as with an array. +Both iterables and array-likes are usually *not arrays*, they don't have `push`, `pop` etc. That's rather inconvenient if we have such an object and want to work with it as with an array. E.g. we would like to work with `range` using array methods. How to achieve that? ## Array.from -There's a universal method [Array.from](mdn:js/Array/from) that brings them together. It takes an iterable or array-like value and makes a "real" `Array` from it. Then we can call array methods on it. +There's a universal method [Array.from](mdn:js/Array/from) that takes an iterable or array-like value and makes a "real" `Array` from it. Then we can call array methods on it. For instance: @@ -212,26 +214,26 @@ let arr = Array.from(arrayLike); // (*) alert(arr.pop()); // World (method works) ``` -`Array.from` at the line `(*)` takes the object, examines it for being an iterable or array-like, then makes a new array and copies there all items. +`Array.from` at the line `(*)` takes the object, examines it for being an iterable or array-like, then makes a new array and copies all items to it. The same happens for an iterable: -```js +```js run // assuming that range is taken from the example above let arr = Array.from(range); alert(arr); // 1,2,3,4,5 (array toString conversion works) ``` -The full syntax for `Array.from` allows to provide an optional "mapping" function: +The full syntax for `Array.from` also allows us to provide an optional "mapping" function: ```js Array.from(obj[, mapFn, thisArg]) ``` -The second argument `mapFn` should be the function to apply to each element before adding to the array, and `thisArg` allows to set `this` for it. +The optional second argument `mapFn` can be a function that will be applied to each element before adding it to the array, and `thisArg` allows us to set `this` for it. For instance: -```js +```js run // assuming that range is taken from the example above // square each number @@ -268,7 +270,7 @@ for (let char of str) { alert(chars); ``` -...But is shorter. +...But it is shorter. We can even build surrogate-aware `slice` on it: @@ -281,7 +283,7 @@ let str = '𝒳😂𩷶'; alert( slice(str, 1, 3) ); // 😂𩷶 -// native method does not support surrogate pairs +// the native method does not support surrogate pairs alert( str.slice(1, 3) ); // garbage (two pieces from different surrogate pairs) ``` @@ -291,8 +293,8 @@ alert( str.slice(1, 3) ); // garbage (two pieces from different surrogate pairs) Objects that can be used in `for..of` are called *iterable*. - Technically, iterables must implement the method named `Symbol.iterator`. - - The result of `obj[Symbol.iterator]` is called an *iterator*. It handles the further iteration process. - - An iterator must have the method named `next()` that returns an object `{done: Boolean, value: any}`, here `done:true` denotes the iteration end, otherwise the `value` is the next value. + - The result of `obj[Symbol.iterator]()` is called an *iterator*. It handles further iteration process. + - An iterator must have the method named `next()` that returns an object `{done: Boolean, value: any}`, here `done:true` denotes the end of the iteration process, otherwise the `value` is the next value. - The `Symbol.iterator` method is called automatically by `for..of`, but we also can do it directly. - Built-in iterables like strings or arrays, also implement `Symbol.iterator`. - String iterator knows about surrogate pairs. @@ -302,4 +304,4 @@ Objects that have indexed properties and `length` are called *array-like*. Such If we look inside the specification -- we'll see that most built-in methods assume that they work with iterables or array-likes instead of "real" arrays, because that's more abstract. -`Array.from(obj[, mapFn, thisArg])` makes a real `Array` of an iterable or array-like `obj`, and we can then use array methods on it. The optional arguments `mapFn` and `thisArg` allow us to apply a function to each item. +`Array.from(obj[, mapFn, thisArg])` makes a real `Array` from an iterable or array-like `obj`, and we can then use array methods on it. The optional arguments `mapFn` and `thisArg` allow us to apply a function to each item. diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/solution.md b/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/solution.md deleted file mode 100644 index ce56f593a..000000000 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/solution.md +++ /dev/null @@ -1,41 +0,0 @@ -The sane choice here is a `WeakSet`: - -```js -let messages = [ - {text: "Hello", from: "John"}, - {text: "How goes?", from: "John"}, - {text: "See you soon", from: "Alice"} -]; - -let readMessages = new WeakSet(); - -// two messages have been read -readMessages.add(messages[0]); -readMessages.add(messages[1]); -// readMessages has 2 elements - -// ...let's read the first message again! -readMessages.add(messages[0]); -// readMessages still has 2 unique elements - -// answer: was the message[0] read? -alert("Read message 0: " + readMessages.has(messages[0])); // true - -messages.shift(); -// now readMessages has 1 element (technically memory may be cleaned later) -``` - -The `WeakSet` allows to store a set of messages and easily check for the existance of a message in it. - -It cleans up itself automatically. The tradeoff is that we can't iterate over it. We can't get "all read messages" directly. But we can do it by iterating over all messages and filtering those that are in the set. - -P.S. Adding a property of our own to each message may be dangerous if messages are managed by someone else's code, but we can make it a symbol to evade conflicts. - -Like this: -```js -// the symbolic property is only known to our code -let isRead = Symbol("isRead"); -messages[0][isRead] = true; -``` - -Now even if someone else's code uses `for..in` loop for message properties, our secret flag won't appear. diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/task.md b/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/task.md deleted file mode 100644 index 7ec1faf16..000000000 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/04-recipients-read/task.md +++ /dev/null @@ -1,23 +0,0 @@ -importance: 5 - ---- - -# Store "unread" flags - -There's an array of messages: - -```js -let messages = [ - {text: "Hello", from: "John"}, - {text: "How goes?", from: "John"}, - {text: "See you soon", from: "Alice"} -]; -``` - -Your code can access it, but the messages are managed by someone else's code. New messages are added, old ones are removed regularly by that code, and you don't know the exact moments when it happens. - -Now, which data structure you could use to store information whether the message "have been read"? The structure must be well-suited to give the answer "was it read?" for the given message object. - -P.S. When a message is removed from `messages`, it should disappear from your structure as well. - -P.P.S. We shouldn't modify message objects directly. If they are managed by someone else's code, then adding extra properties to them may have bad consequences. diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/article.md b/1-js/05-data-types/07-map-set-weakmap-weakset/article.md deleted file mode 100644 index 034ad22c3..000000000 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/article.md +++ /dev/null @@ -1,458 +0,0 @@ - -# Map, Set, WeakMap and WeakSet - -Now we've learned about the following complex data structures: - -- Objects for storing keyed collections. -- Arrays for storing ordered collections. - -But that's not enough for real life. That's why `Map` and `Set` also exist. - -## Map - -[Map](mdn:js/Map) is a collection of keyed data items, just like an `Object`. But the main difference is that `Map` allows keys of any type. - -The main methods are: - -- `new Map()` -- creates the map. -- `map.set(key, value)` -- stores the value by the key. -- `map.get(key)` -- returns the value by the key, `undefined` if `key` doesn't exist in map. -- `map.has(key)` -- returns `true` if the `key` exists, `false` otherwise. -- `map.delete(key)` -- removes the value by the key. -- `map.clear()` -- clears the map -- `map.size` -- returns the current element count. - -For instance: - -```js run -let map = new Map(); - -map.set('1', 'str1'); // a string key -map.set(1, 'num1'); // a numeric key -map.set(true, 'bool1'); // a boolean key - -// remember the regular Object? it would convert keys to string -// Map keeps the type, so these two are different: -alert( map.get(1) ); // 'num1' -alert( map.get('1') ); // 'str1' - -alert( map.size ); // 3 -``` - -As we can see, unlike objects, keys are not converted to strings. Any type of key is possible. - -**Map can also use objects as keys.** - -For instance: -```js run -let john = { name: "John" }; - -// for every user, let's store their visits count -let visitsCountMap = new Map(); - -// john is the key for the map -visitsCountMap.set(john, 123); - -alert( visitsCountMap.get(john) ); // 123 -``` - -Using objects as keys is one of most notable and important `Map` features. For string keys, `Object` can be fine, but it would be difficult to replace the `Map` with a regular `Object` in the example above. - -In the old times, before `Map` existed, people added unique identifiers to objects for that: - -```js run -// we add the id field -let john = { name: "John", *!*id: 1*/!* }; - -let visitsCounts = {}; - -// now store the value by id -visitsCounts[john.id] = 123; - -alert( visitsCounts[john.id] ); // 123 -``` - -...But `Map` is much more elegant. - - -```smart header="How `Map` compares keys" -To test values for equivalence, `Map` uses the algorithm [SameValueZero](https://tc39.github.io/ecma262/#sec-samevaluezero). It is roughly the same as strict equality `===`, but the difference is that `NaN` is considered equal to `NaN`. So `NaN` can be used as the key as well. - -This algorithm can't be changed or customized. -``` - - -````smart header="Chaining" - -Every `map.set` call returns the map itself, so we can "chain" the calls: - -```js -map.set('1', 'str1') - .set(1, 'num1') - .set(true, 'bool1'); -``` -```` - -## Map from Object - -When a `Map` is created, we can pass an array (or another iterable) with key-value pairs, like this: - -```js -// array of [key, value] pairs -let map = new Map([ - ['1', 'str1'], - [1, 'num1'], - [true, 'bool1'] -]); -``` - -There is a built-in method [Object.entries(obj)](mdn:js/Object/entries) that returns an array of key/value pairs for an object exactly in that format. - -So we can initialize a map from an object like this: - -```js -let map = new Map(Object.entries({ - name: "John", - age: 30 -})); -``` - -Here, `Object.entries` returns the array of key/value pairs: `[ ["name","John"], ["age", 30] ]`. That's what `Map` needs. - -## Iteration over Map - -For looping over a `map`, there are 3 methods: - -- `map.keys()` -- returns an iterable for keys, -- `map.values()` -- returns an iterable for values, -- `map.entries()` -- returns an iterable for entries `[key, value]`, it's used by default in `for..of`. - -For instance: - -```js run -let recipeMap = new Map([ - ['cucumber', 500], - ['tomatoes', 350], - ['onion', 50] -]); - -// iterate over keys (vegetables) -for (let vegetable of recipeMap.keys()) { - alert(vegetable); // cucumber, tomatoes, onion -} - -// iterate over values (amounts) -for (let amount of recipeMap.values()) { - alert(amount); // 500, 350, 50 -} - -// iterate over [key, value] entries -for (let entry of recipeMap) { // the same as of recipeMap.entries() - alert(entry); // cucumber,500 (and so on) -} -``` - -```smart header="The insertion order is used" -The iteration goes in the same order as the values were inserted. `Map` preserves this order, unlike a regular `Object`. -``` - -Besides that, `Map` has a built-in `forEach` method, similar to `Array`: - -```js -// runs the function for each (key, value) pair -recipeMap.forEach( (value, key, map) => { - alert(`${key}: ${value}`); // cucumber: 500 etc -}); -``` - - -## Set - -A `Set` is a collection of values, where each value may occur only once. - -Its main methods are: - -- `new Set(iterable)` -- creates the set, optionally from an array of values (any iterable will do). -- `set.add(value)` -- adds a value, returns the set itself. -- `set.delete(value)` -- removes the value, returns `true` if `value` existed at the moment of the call, otherwise `false`. -- `set.has(value)` -- returns `true` if the value exists in the set, otherwise `false`. -- `set.clear()` -- removes everything from the set. -- `set.size` -- is the elements count. - -For example, we have visitors coming, and we'd like to remember everyone. But repeated visits should not lead to duplicates. A visitor must be "counted" only once. - -`Set` is just the right thing for that: - -```js run -let set = new Set(); - -let john = { name: "John" }; -let pete = { name: "Pete" }; -let mary = { name: "Mary" }; - -// visits, some users come multiple times -set.add(john); -set.add(pete); -set.add(mary); -set.add(john); -set.add(mary); - -// set keeps only unique values -alert( set.size ); // 3 - -for (let user of set) { - alert(user.name); // John (then Pete and Mary) -} -``` - -The alternative to `Set` could be an array of users, and the code to check for duplicates on every insertion using [arr.find](mdn:js/Array/find). But the performance would be much worse, because this method walks through the whole array checking every element. `Set` is much better optimized internally for uniqueness checks. - -## Iteration over Set - -We can loop over a set either with `for..of` or using `forEach`: - -```js run -let set = new Set(["oranges", "apples", "bananas"]); - -for (let value of set) alert(value); - -// the same with forEach: -set.forEach((value, valueAgain, set) => { - alert(value); -}); -``` - -Note the funny thing. The `forEach` function in the `Set` has 3 arguments: a value, then *again a value*, and then the target object. Indeed, the same value appears in the arguments twice. - -That's for compatibility with `Map` where `forEach` has three arguments. Looks a bit strange, for sure. But may help to replace `Map` with `Set` in certain cases with ease, and vice versa. - -The same methods `Map` has for iterators are also supported: - -- `set.keys()` -- returns an iterable object for values, -- `set.values()` -- same as `set.keys`, for compatibility with `Map`, -- `set.entries()` -- returns an iterable object for entries `[value, value]`, exists for compatibility with `Map`. - -## WeakMap and WeakSet - -`WeakSet` is a special kind of `Set` that does not prevent JavaScript from removing its items from memory. `WeakMap` is the same thing for `Map`. - -As we know from the chapter <info:garbage-collection>, JavaScript engine stores a value in memory while it is reachable (and can potentially be used). - -For instance: -```js -let john = { name: "John" }; - -// the object can be accessed, john is the reference to it - -// overwrite the reference -john = null; - -*!* -// the object will be removed from memory -*/!* -``` - -Usually, properties of an object or elements of an array or another data structure are considered reachable and kept in memory while that data structure is in memory. - -For instance, if we put an object into an array, then while the array is alive, the object will be alive as well, even if there are no other references to it. - -Like this: - -```js -let john = { name: "John" }; - -let array = [ john ]; - -john = null; // overwrite the reference - -*!* -// john is stored inside the array, so it won't be garbage-collected -// we can get it as array[0] -*/!* -``` - -Or, if we use an object as the key in a regular `Map`, then while the `Map` exists, that object exists as well. It occupies memory and may not be garbage collected. - -For instance: - -```js -let john = { name: "John" }; - -let map = new Map(); -map.set(john, "..."); - -john = null; // overwrite the reference - -*!* -// john is stored inside the map, -// we can get it by using map.keys() -*/!* -``` - -`WeakMap/WeakSet` are fundamentally different in this aspect. They do not prevent garbage-collection of key objects. - -Let's explain it starting with `WeakMap`. - -The first difference from `Map` is that `WeakMap` keys must be objects, not primitive values: - -```js run -let weakMap = new WeakMap(); - -let obj = {}; - -weakMap.set(obj, "ok"); // works fine (object key) - -*!* -// can't use a string as the key -weakMap.set("test", "Whoops"); // Error, because "test" is not an object -*/!* -``` - -Now, if we use an object as the key in it, and there are no other references to that object -- it will be removed from memory (and from the map) automatically. - -```js -let john = { name: "John" }; - -let weakMap = new WeakMap(); -weakMap.set(john, "..."); - -john = null; // overwrite the reference - -// john is removed from memory! -``` - -Compare it with the regular `Map` example above. Now if `john` only exists as the key of `WeakMap` -- it is to be automatically deleted. - -`WeakMap` does not support iteration and methods `keys()`, `values()`, `entries()`, so there's no way to get all keys or values from it. - -`WeakMap` has only the following methods: - -- `weakMap.get(key)` -- `weakMap.set(key, value)` -- `weakMap.delete(key)` -- `weakMap.has(key)` - -Why such a limitation? That's for technical reasons. If an object has lost all other references (like `john` in the code above), then it is to be garbage-collected automatically. But technically it's not exactly specified *when the cleanup happens*. - -The JavaScript engine decides that. It may choose to perform the memory cleanup immediately or to wait and do the cleaning later when more deletions happen. So, technically the current element count of a `WeakMap` is not known. The engine may have cleaned it up or not, or did it partially. For that reason, methods that access `WeakMap` as a whole are not supported. - -Now where do we need such thing? - -The idea of `WeakMap` is that we can store something for an object that should exist only while the object exists. But we do not force the object to live by the mere fact that we store something for it. - -```js -weakMap.set(john, "secret documents"); -// if john dies, secret documents will be destroyed automatically -``` - -That's useful for situations when we have a main storage for the objects somewhere and need to keep additional information, that is only relevant while the object lives. - -Let's look at an example. - -For instance, we have code that keeps a visit count for each user. The information is stored in a map: a user is the key and the visit count is the value. When a user leaves, we don't want to store their visit count anymore. - -One way would be to keep track of users, and when they leave -- clean up the map manually: - -```js run -let john = { name: "John" }; - -// map: user => visits count -let visitsCountMap = new Map(); - -// john is the key for the map -visitsCountMap.set(john, 123); - -// now john leaves us, we don't need him anymore -john = null; - -*!* -// but it's still in the map, we need to clean it! -*/!* -alert( visitsCountMap.size ); // 1 -// and john is also in the memory, because Map uses it as the key -``` - -Another way would be to use `WeakMap`: - -```js -let john = { name: "John" }; - -let visitsCountMap = new WeakMap(); - -visitsCountMap.set(john, 123); - -// now john leaves us, we don't need him anymore -john = null; - -// there are no references except WeakMap, -// so the object is removed both from the memory and from visitsCountMap automatically -``` - -With a regular `Map`, cleaning up after a user has left becomes a tedious task: we not only need to remove the user from its main storage (be it a variable or an array), but also need to clean up the additional stores like `visitsCountMap`. And it can become cumbersome in more complex cases when users are managed in one place of the code and the additional structure is in another place and is getting no information about removals. - -```summary -`WeakMap` can make things simpler, because it is cleaned up automatically. The information in it like visits count in the example above lives only while the key object exists. -``` - -`WeakSet` behaves similarly: - -- It is analogous to `Set`, but we may only add objects to `WeakSet` (not primitives). -- An object exists in the set while it is reachable from somewhere else. -- Like `Set`, it supports `add`, `has` and `delete`, but not `size`, `keys()` and no iterations. - -For instance, we can use it to keep track of whether a message is read: - -```js -let messages = [ - {text: "Hello", from: "John"}, - {text: "How goes?", from: "John"}, - {text: "See you soon", from: "Alice"} -]; - -// fill it with array elements (3 items) -let unreadSet = new WeakSet(messages); - -// use unreadSet to see whether a message is unread -alert(unreadSet.has(messages[1])); // true - -// remove it from the set after reading -unreadSet.delete(messages[1]); // true - -// and when we shift our messages history, the set is cleaned up automatically -messages.shift(); - -*!* -// no need to clean unreadSet, it now has 2 items -*/!* -// (though technically we don't know for sure when the JS engine clears it) -``` - -The most notable limitation of `WeakMap` and `WeakSet` is the absence of iterations, and inability to get all current content. That may appear inconvenient, but does not prevent `WeakMap/WeakSet` from doing their main job -- be an "additional" storage of data for objects which are stored/managed at another place. - -## Summary - -Regular collections: -- `Map` -- is a collection of keyed values. - - The differences from a regular `Object`: - - - Any keys, objects can be keys. - - Iterates in the insertion order. - - Additional convenient methods, the `size` property. - -- `Set` -- is a collection of unique values. - - - Unlike an array, does not allow to reorder elements. - - Keeps the insertion order. - -Collections that allow garbage-collection: - -- `WeakMap` -- a variant of `Map` that allows only objects as keys and removes them once they become inaccessible by other means. - - - It does not support operations on the structure as a whole: no `size`, no `clear()`, no iterations. - -- `WeakSet` -- is a variant of `Set` that only stores objects and removes them once they become inaccessible by other means. - - - Also does not support `size/clear()` and iterations. - -`WeakMap` and `WeakSet` are used as "secondary" data structures in addition to the "main" object storage. Once the object is removed from the main storage, if it is only found in the `WeakMap/WeakSet`, it will be cleaned up automatically. diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/_js.view/solution.js b/1-js/05-data-types/07-map-set/01-array-unique-map/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/_js.view/solution.js rename to 1-js/05-data-types/07-map-set/01-array-unique-map/_js.view/solution.js diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/_js.view/test.js b/1-js/05-data-types/07-map-set/01-array-unique-map/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/_js.view/test.js rename to 1-js/05-data-types/07-map-set/01-array-unique-map/_js.view/test.js diff --git a/1-js/05-data-types/08-keys-values-entries/02-count-properties/solution.md b/1-js/05-data-types/07-map-set/01-array-unique-map/solution.md similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/02-count-properties/solution.md rename to 1-js/05-data-types/07-map-set/01-array-unique-map/solution.md diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/task.md b/1-js/05-data-types/07-map-set/01-array-unique-map/task.md similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/01-array-unique-map/task.md rename to 1-js/05-data-types/07-map-set/01-array-unique-map/task.md diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/_js.view/solution.js b/1-js/05-data-types/07-map-set/02-filter-anagrams/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/_js.view/solution.js rename to 1-js/05-data-types/07-map-set/02-filter-anagrams/_js.view/solution.js diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/_js.view/test.js b/1-js/05-data-types/07-map-set/02-filter-anagrams/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/_js.view/test.js rename to 1-js/05-data-types/07-map-set/02-filter-anagrams/_js.view/test.js diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/solution.md b/1-js/05-data-types/07-map-set/02-filter-anagrams/solution.md similarity index 98% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/solution.md rename to 1-js/05-data-types/07-map-set/02-filter-anagrams/solution.md index 4c8af1f24..160675185 100644 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/solution.md +++ b/1-js/05-data-types/07-map-set/02-filter-anagrams/solution.md @@ -36,7 +36,7 @@ Letter-sorting is done by the chain of calls in the line `(*)`. For convenience let's split it into multiple lines: ```js -let sorted = arr[i] // PAN +let sorted = word // PAN .toLowerCase() // pan .split('') // ['p','a','n'] .sort() // ['a','n','p'] diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/task.md b/1-js/05-data-types/07-map-set/02-filter-anagrams/task.md similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/02-filter-anagrams/task.md rename to 1-js/05-data-types/07-map-set/02-filter-anagrams/task.md diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/03-iterable-keys/solution.md b/1-js/05-data-types/07-map-set/03-iterable-keys/solution.md similarity index 100% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/03-iterable-keys/solution.md rename to 1-js/05-data-types/07-map-set/03-iterable-keys/solution.md diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/03-iterable-keys/task.md b/1-js/05-data-types/07-map-set/03-iterable-keys/task.md similarity index 64% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/03-iterable-keys/task.md rename to 1-js/05-data-types/07-map-set/03-iterable-keys/task.md index b1ccbd0ac..81507647f 100644 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/03-iterable-keys/task.md +++ b/1-js/05-data-types/07-map-set/03-iterable-keys/task.md @@ -4,9 +4,9 @@ importance: 5 # Iterable keys -We want to get an array of `map.keys()` and go on working with it (apart from the map itself). +We'd like to get an array of `map.keys()` in a variable and then apply array-specific methods to it, e.g. `.push`. -But there's a problem: +But that doesn't work: ```js run let map = new Map(); diff --git a/1-js/05-data-types/07-map-set/article.md b/1-js/05-data-types/07-map-set/article.md new file mode 100644 index 000000000..37f5e48c2 --- /dev/null +++ b/1-js/05-data-types/07-map-set/article.md @@ -0,0 +1,331 @@ + +# Map and Set + +Till now, we've learned about the following complex data structures: + +- Objects are used for storing keyed collections. +- Arrays are used for storing ordered collections. + +But that's not enough for real life. That's why `Map` and `Set` also exist. + +## Map + +[Map](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) is a collection of keyed data items, just like an `Object`. But the main difference is that `Map` allows keys of any type. + +Methods and properties are: + +- [`new Map()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/Map) -- creates the map. +- [`map.set(key, value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/set) -- stores the value by the key. +- [`map.get(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/get) -- returns the value by the key, `undefined` if `key` doesn't exist in map. +- [`map.has(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/has) -- returns `true` if the `key` exists, `false` otherwise. +- [`map.delete(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/delete) -- removes the element (the key/value pair) by the key. +- [`map.clear()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/clear) -- removes everything from the map. +- [`map.size`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/size) -- returns the current element count. + +For instance: + +```js run +let map = new Map(); + +map.set('1', 'str1'); // a string key +map.set(1, 'num1'); // a numeric key +map.set(true, 'bool1'); // a boolean key + +// remember the regular Object? it would convert keys to string +// Map keeps the type, so these two are different: +alert( map.get(1) ); // 'num1' +alert( map.get('1') ); // 'str1' + +alert( map.size ); // 3 +``` + +As we can see, unlike objects, keys are not converted to strings. Any type of key is possible. + +```smart header="`map[key]` isn't the right way to use a `Map`" +Although `map[key]` also works, e.g. we can set `map[key] = 2`, this is treating `map` as a plain JavaScript object, so it implies all corresponding limitations (only string/symbol keys and so on). + +So we should use `map` methods: `set`, `get` and so on. +``` + +**Map can also use objects as keys.** + +For instance: + +```js run +let john = { name: "John" }; + +// for every user, let's store their visits count +let visitsCountMap = new Map(); + +// john is the key for the map +visitsCountMap.set(john, 123); + +alert( visitsCountMap.get(john) ); // 123 +``` + +Using objects as keys is one of the most notable and important `Map` features. The same does not count for `Object`. String as a key in `Object` is fine, but we can't use another `Object` as a key in `Object`. + +Let's try: + +```js run +let john = { name: "John" }; +let ben = { name: "Ben" }; + +let visitsCountObj = {}; // try to use an object + +visitsCountObj[ben] = 234; // try to use ben object as the key +visitsCountObj[john] = 123; // try to use john object as the key, ben object will get replaced + +*!* +// That's what got written! +alert( visitsCountObj["[object Object]"] ); // 123 +*/!* +``` + +As `visitsCountObj` is an object, it converts all `Object` keys, such as `john` and `ben` above, to same string `"[object Object]"`. Definitely not what we want. + +```smart header="How `Map` compares keys" +To test keys for equivalence, `Map` uses the algorithm [SameValueZero](https://tc39.github.io/ecma262/#sec-samevaluezero). It is roughly the same as strict equality `===`, but the difference is that `NaN` is considered equal to `NaN`. So `NaN` can be used as the key as well. + +This algorithm can't be changed or customized. +``` + +````smart header="Chaining" +Every `map.set` call returns the map itself, so we can "chain" the calls: + +```js +map.set('1', 'str1') + .set(1, 'num1') + .set(true, 'bool1'); +``` +```` + +## Iteration over Map + +For looping over a `map`, there are 3 methods: + +- [`map.keys()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/keys) -- returns an iterable for keys, +- [`map.values()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/values) -- returns an iterable for values, +- [`map.entries()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/entries) -- returns an iterable for entries `[key, value]`, it's used by default in `for..of`. + +For instance: + +```js run +let recipeMap = new Map([ + ['cucumber', 500], + ['tomatoes', 350], + ['onion', 50] +]); + +// iterate over keys (vegetables) +for (let vegetable of recipeMap.keys()) { + alert(vegetable); // cucumber, tomatoes, onion +} + +// iterate over values (amounts) +for (let amount of recipeMap.values()) { + alert(amount); // 500, 350, 50 +} + +// iterate over [key, value] entries +for (let entry of recipeMap) { // the same as of recipeMap.entries() + alert(entry); // cucumber,500 (and so on) +} +``` + +```smart header="The insertion order is used" +The iteration goes in the same order as the values were inserted. `Map` preserves this order, unlike a regular `Object`. +``` + +Besides that, `Map` has a built-in `forEach` method, similar to `Array`: + +```js +// runs the function for each (key, value) pair +recipeMap.forEach( (value, key, map) => { + alert(`${key}: ${value}`); // cucumber: 500 etc +}); +``` + +## Object.entries: Map from Object + +When a `Map` is created, we can pass an array (or another iterable) with key/value pairs for initialization, like this: + +```js run +// array of [key, value] pairs +let map = new Map([ + ['1', 'str1'], + [1, 'num1'], + [true, 'bool1'] +]); + +alert( map.get('1') ); // str1 +``` + +If we have a plain object, and we'd like to create a `Map` from it, then we can use built-in method [Object.entries(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/entries) that returns an array of key/value pairs for an object exactly in that format. + +So we can create a map from an object like this: + +```js run +let obj = { + name: "John", + age: 30 +}; + +*!* +let map = new Map(Object.entries(obj)); +*/!* + +alert( map.get('name') ); // John +``` + +Here, `Object.entries` returns the array of key/value pairs: `[ ["name","John"], ["age", 30] ]`. That's what `Map` needs. + + +## Object.fromEntries: Object from Map + +We've just seen how to create `Map` from a plain object with `Object.entries(obj)`. + +There's `Object.fromEntries` method that does the reverse: given an array of `[key, value]` pairs, it creates an object from them: + +```js run +let prices = Object.fromEntries([ + ['banana', 1], + ['orange', 2], + ['meat', 4] +]); + +// now prices = { banana: 1, orange: 2, meat: 4 } + +alert(prices.orange); // 2 +``` + +We can use `Object.fromEntries` to get a plain object from `Map`. + +E.g. we store the data in a `Map`, but we need to pass it to a 3rd-party code that expects a plain object. + +Here we go: + +```js run +let map = new Map(); +map.set('banana', 1); +map.set('orange', 2); +map.set('meat', 4); + +*!* +let obj = Object.fromEntries(map.entries()); // make a plain object (*) +*/!* + +// done! +// obj = { banana: 1, orange: 2, meat: 4 } + +alert(obj.orange); // 2 +``` + +A call to `map.entries()` returns an iterable of key/value pairs, exactly in the right format for `Object.fromEntries`. + +We could also make line `(*)` shorter: +```js +let obj = Object.fromEntries(map); // omit .entries() +``` + +That's the same, because `Object.fromEntries` expects an iterable object as the argument. Not necessarily an array. And the standard iteration for `map` returns same key/value pairs as `map.entries()`. So we get a plain object with same key/values as the `map`. + +## Set + +A [`Set`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set) is a special type collection - "set of values" (without keys), where each value may occur only once. + +Its main methods are: + +- [`new Set([iterable])`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/Set) -- creates the set, and if an `iterable` object is provided (usually an array), copies values from it into the set. +- [`set.add(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/add) -- adds a value, returns the set itself. +- [`set.delete(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/delete) -- removes the value, returns `true` if `value` existed at the moment of the call, otherwise `false`. +- [`set.has(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/has) -- returns `true` if the value exists in the set, otherwise `false`. +- [`set.clear()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/clear) -- removes everything from the set. +- [`set.size`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/size) -- is the elements count. + +The main feature is that repeated calls of `set.add(value)` with the same value don't do anything. That's the reason why each value appears in a `Set` only once. + +For example, we have visitors coming, and we'd like to remember everyone. But repeated visits should not lead to duplicates. A visitor must be "counted" only once. + +`Set` is just the right thing for that: + +```js run +let set = new Set(); + +let john = { name: "John" }; +let pete = { name: "Pete" }; +let mary = { name: "Mary" }; + +// visits, some users come multiple times +set.add(john); +set.add(pete); +set.add(mary); +set.add(john); +set.add(mary); + +// set keeps only unique values +alert( set.size ); // 3 + +for (let user of set) { + alert(user.name); // John (then Pete and Mary) +} +``` + +The alternative to `Set` could be an array of users, and the code to check for duplicates on every insertion using [arr.find](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find). But the performance would be much worse, because this method walks through the whole array checking every element. `Set` is much better optimized internally for uniqueness checks. + +## Iteration over Set + +We can loop over a set either with `for..of` or using `forEach`: + +```js run +let set = new Set(["oranges", "apples", "bananas"]); + +for (let value of set) alert(value); + +// the same with forEach: +set.forEach((value, valueAgain, set) => { + alert(value); +}); +``` + +Note the funny thing. The callback function passed in `forEach` has 3 arguments: a `value`, then *the same value* `valueAgain`, and then the target object. Indeed, the same value appears in the arguments twice. + +That's for compatibility with `Map` where the callback passed `forEach` has three arguments. Looks a bit strange, for sure. But this may help to replace `Map` with `Set` in certain cases with ease, and vice versa. + +The same methods `Map` has for iterators are also supported: + +- [`set.keys()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/keys) -- returns an iterable object for values, +- [`set.values()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/values) -- same as `set.keys()`, for compatibility with `Map`, +- [`set.entries()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/entries) -- returns an iterable object for entries `[value, value]`, exists for compatibility with `Map`. + +## Summary + +[`Map`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) -- is a collection of keyed values. + +Methods and properties: + +- [`new Map([iterable])`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/Map) -- creates the map, with optional `iterable` (e.g. array) of `[key,value]` pairs for initialization. +- [`map.set(key, value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/set) -- stores the value by the key, returns the map itself. +- [`map.get(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/get) -- returns the value by the key, `undefined` if `key` doesn't exist in map. +- [`map.has(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/has) -- returns `true` if the `key` exists, `false` otherwise. +- [`map.delete(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/delete) -- removes the element by the key, returns `true` if `key` existed at the moment of the call, otherwise `false`. +- [`map.clear()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/clear) -- removes everything from the map. +- [`map.size`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map/size) -- returns the current element count. + +The differences from a regular `Object`: + +- Any keys, objects can be keys. +- Additional convenient methods, the `size` property. + +[`Set`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set) -- is a collection of unique values. + +Methods and properties: + +- [`new Set([iterable])`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/Set) -- creates the set, with optional `iterable` (e.g. array) of values for initialization. +- [`set.add(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/add) -- adds a value (does nothing if `value` exists), returns the set itself. +- [`set.delete(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/delete) -- removes the value, returns `true` if `value` existed at the moment of the call, otherwise `false`. +- [`set.has(value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/has) -- returns `true` if the value exists in the set, otherwise `false`. +- [`set.clear()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/clear) -- removes everything from the set. +- [`set.size`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set/size) -- is the elements count. + +Iteration over `Map` and `Set` is always in the insertion order, so we can't say that these collections are unordered, but we can't reorder elements or directly get an element by its number. diff --git a/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/solution.md b/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/solution.md new file mode 100644 index 000000000..e2147ccfa --- /dev/null +++ b/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/solution.md @@ -0,0 +1,43 @@ +Let's store read messages in `WeakSet`: + +```js run +let messages = [ + {text: "Hello", from: "John"}, + {text: "How goes?", from: "John"}, + {text: "See you soon", from: "Alice"} +]; + +let readMessages = new WeakSet(); + +// two messages have been read +readMessages.add(messages[0]); +readMessages.add(messages[1]); +// readMessages has 2 elements + +// ...let's read the first message again! +readMessages.add(messages[0]); +// readMessages still has 2 unique elements + +// answer: was the message[0] read? +alert("Read message 0: " + readMessages.has(messages[0])); // true + +messages.shift(); +// now readMessages has 1 element (technically memory may be cleaned later) +``` + +The `WeakSet` allows to store a set of messages and easily check for the existence of a message in it. + +It cleans up itself automatically. The tradeoff is that we can't iterate over it, can't get "all read messages" from it directly. But we can do it by iterating over all messages and filtering those that are in the set. + +Another, different solution could be to add a property like `message.isRead=true` to a message after it's read. As messages objects are managed by another code, that's generally discouraged, but we can use a symbolic property to avoid conflicts. + +Like this: +```js +// the symbolic property is only known to our code +let isRead = Symbol("isRead"); +messages[0][isRead] = true; +``` + +Now third-party code probably won't see our extra property. + +Although symbols allow to lower the probability of problems, using `WeakSet` is better from the architectural point of view. diff --git a/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/task.md b/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/task.md new file mode 100644 index 000000000..fd31a891b --- /dev/null +++ b/1-js/05-data-types/08-weakmap-weakset/01-recipients-read/task.md @@ -0,0 +1,23 @@ +importance: 5 + +--- + +# Store "unread" flags + +There's an array of messages: + +```js +let messages = [ + {text: "Hello", from: "John"}, + {text: "How goes?", from: "John"}, + {text: "See you soon", from: "Alice"} +]; +``` + +Your code can access it, but the messages are managed by someone else's code. New messages are added, old ones are removed regularly by that code, and you don't know the exact moments when it happens. + +Now, which data structure could you use to store information about whether the message "has been read"? The structure must be well-suited to give the answer "was it read?" for the given message object. + +P.S. When a message is removed from `messages`, it should disappear from your structure as well. + +P.P.S. We shouldn't modify message objects, add our properties to them. As they are managed by someone else's code, that may lead to bad consequences. diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/solution.md b/1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/solution.md similarity index 61% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/solution.md rename to 1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/solution.md index 7f387b4da..2af0547c1 100644 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/solution.md +++ b/1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/solution.md @@ -3,9 +3,9 @@ To store a date, we can use `WeakMap`: ```js let messages = [ - {text: "Hello", from: "John"}, - {text: "How goes?", from: "John"}, - {text: "See you soon", from: "Alice"} + {text: "Hello", from: "John"}, + {text: "How goes?", from: "John"}, + {text: "See you soon", from: "Alice"} ]; let readMap = new WeakMap(); diff --git a/1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/task.md b/1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/task.md similarity index 54% rename from 1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/task.md rename to 1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/task.md index 22b51a382..8e341c184 100644 --- a/1-js/05-data-types/07-map-set-weakmap-weakset/05-recipients-when-read/task.md +++ b/1-js/05-data-types/08-weakmap-weakset/02-recipients-when-read/task.md @@ -8,12 +8,14 @@ There's an array of messages as in the [previous task](info:task/recipients-read ```js let messages = [ - {text: "Hello", from: "John"}, - {text: "How goes?", from: "John"}, - {text: "See you soon", from: "Alice"} + {text: "Hello", from: "John"}, + {text: "How goes?", from: "John"}, + {text: "See you soon", from: "Alice"} ]; ``` The question now is: which data structure you'd suggest to store the information: "when the message was read?". -In the previous task we only needed to store the "yes/no" fact. Now we need to store the date and it, once again, should disappear if the message is gone. +In the previous task we only needed to store the "yes/no" fact. Now we need to store the date, and it should only remain in memory until the message is garbage collected. + +P.S. Dates can be stored as objects of built-in `Date` class, that we'll cover later. diff --git a/1-js/05-data-types/08-weakmap-weakset/article.md b/1-js/05-data-types/08-weakmap-weakset/article.md new file mode 100644 index 000000000..9795017d4 --- /dev/null +++ b/1-js/05-data-types/08-weakmap-weakset/article.md @@ -0,0 +1,295 @@ + +# WeakMap and WeakSet + +As we know from the chapter <info:garbage-collection>, JavaScript engine keeps a value in memory while it is "reachable" and can potentially be used. + +For instance: + +```js +let john = { name: "John" }; + +// the object can be accessed, john is the reference to it + +// overwrite the reference +john = null; + +*!* +// the object will be removed from memory +*/!* +``` + +Usually, properties of an object or elements of an array or another data structure are considered reachable and kept in memory while that data structure is in memory. + +For instance, if we put an object into an array, then while the array is alive, the object will be alive as well, even if there are no other references to it. + +Like this: + +```js +let john = { name: "John" }; + +let array = [ john ]; + +john = null; // overwrite the reference + +*!* +// the object previously referenced by john is stored inside the array +// therefore it won't be garbage-collected +// we can get it as array[0] +*/!* +``` + +Similar to that, if we use an object as the key in a regular `Map`, then while the `Map` exists, that object exists as well. It occupies memory and may not be garbage collected. + +For instance: + +```js +let john = { name: "John" }; + +let map = new Map(); +map.set(john, "..."); + +john = null; // overwrite the reference + +*!* +// john is stored inside the map, +// we can get it by using map.keys() +*/!* +``` + +[`WeakMap`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap) is fundamentally different in this aspect. It doesn't prevent garbage-collection of key objects. + +Let's see what it means on examples. + +## WeakMap + +The first difference between [`Map`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map) and [`WeakMap`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap) is that keys must be objects, not primitive values: + +```js run +let weakMap = new WeakMap(); + +let obj = {}; + +weakMap.set(obj, "ok"); // works fine (object key) + +*!* +// can't use a string as the key +weakMap.set("test", "Whoops"); // Error, because "test" is not an object +*/!* +``` + +Now, if we use an object as the key in it, and there are no other references to that object -- it will be removed from memory (and from the map) automatically. + +```js +let john = { name: "John" }; + +let weakMap = new WeakMap(); +weakMap.set(john, "..."); + +john = null; // overwrite the reference + +// john is removed from memory! +``` + +Compare it with the regular `Map` example above. Now if `john` only exists as the key of `WeakMap` -- it will be automatically deleted from the map (and memory). + +`WeakMap` does not support iteration and methods `keys()`, `values()`, `entries()`, so there's no way to get all keys or values from it. + +`WeakMap` has only the following methods: + +- [`weakMap.set(key, value)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap/set) +- [`weakMap.get(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap/get) +- [`weakMap.delete(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap/delete) +- [`weakMap.has(key)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap/has) + +Why such a limitation? That's for technical reasons. If an object has lost all other references (like `john` in the code above), then it is to be garbage-collected automatically. But technically it's not exactly specified *when the cleanup happens*. + +The JavaScript engine decides that. It may choose to perform the memory cleanup immediately or to wait and do the cleaning later when more deletions happen. So, technically, the current element count of a `WeakMap` is not known. The engine may have cleaned it up or not, or did it partially. For that reason, methods that access all keys/values are not supported. + +Now, where do we need such a data structure? + +## Use case: additional data + +The main area of application for `WeakMap` is an *additional data storage*. + +If we're working with an object that "belongs" to another code, maybe even a third-party library, and would like to store some data associated with it, that should only exist while the object is alive - then `WeakMap` is exactly what's needed. + +We put the data to a `WeakMap`, using the object as the key, and when the object is garbage collected, that data will automatically disappear as well. + +```js +weakMap.set(john, "secret documents"); +// if john dies, secret documents will be destroyed automatically +``` + +Let's look at an example. + +For instance, we have code that keeps a visit count for users. The information is stored in a map: a user object is the key and the visit count is the value. When a user leaves (its object gets garbage collected), we don't want to store their visit count anymore. + +Here's an example of a counting function with `Map`: + +```js +// 📁 visitsCount.js +let visitsCountMap = new Map(); // map: user => visits count + +// increase the visits count +function countUser(user) { + let count = visitsCountMap.get(user) || 0; + visitsCountMap.set(user, count + 1); +} +``` + +And here's another part of the code, maybe another file using it: + +```js +// 📁 main.js +let john = { name: "John" }; + +countUser(john); // count his visits + +// later john leaves us +john = null; +``` + +Now, `john` object should be garbage collected, but remains in memory, as it's a key in `visitsCountMap`. + +We need to clean `visitsCountMap` when we remove users, otherwise it will grow in memory indefinitely. Such cleaning can become a tedious task in complex architectures. + +We can avoid it by switching to `WeakMap` instead: + +```js +// 📁 visitsCount.js +let visitsCountMap = new WeakMap(); // weakmap: user => visits count + +// increase the visits count +function countUser(user) { + let count = visitsCountMap.get(user) || 0; + visitsCountMap.set(user, count + 1); +} +``` + +Now we don't have to clean `visitsCountMap`. After `john` object becomes unreachable, by all means except as a key of `WeakMap`, it gets removed from memory, along with the information by that key from `WeakMap`. + +## Use case: caching + +Another common example is caching. We can store ("cache") results from a function, so that future calls on the same object can reuse it. + +To achieve that, we can use `Map` (not optimal scenario): + +```js run +// 📁 cache.js +let cache = new Map(); + +// calculate and remember the result +function process(obj) { + if (!cache.has(obj)) { + let result = /* calculations of the result for */ obj; + + cache.set(obj, result); + return result; + } + + return cache.get(obj); +} + +*!* +// Now we use process() in another file: +*/!* + +// 📁 main.js +let obj = {/* let's say we have an object */}; + +let result1 = process(obj); // calculated + +// ...later, from another place of the code... +let result2 = process(obj); // remembered result taken from cache + +// ...later, when the object is not needed any more: +obj = null; + +alert(cache.size); // 1 (Ouch! The object is still in cache, taking memory!) +``` + +For multiple calls of `process(obj)` with the same object, it only calculates the result the first time, and then just takes it from `cache`. The downside is that we need to clean `cache` when the object is not needed any more. + +If we replace `Map` with `WeakMap`, then this problem disappears. The cached result will be removed from memory automatically after the object gets garbage collected. + +```js run +// 📁 cache.js +*!* +let cache = new WeakMap(); +*/!* + +// calculate and remember the result +function process(obj) { + if (!cache.has(obj)) { + let result = /* calculate the result for */ obj; + + cache.set(obj, result); + return result; + } + + return cache.get(obj); +} + +// 📁 main.js +let obj = {/* some object */}; + +let result1 = process(obj); +let result2 = process(obj); + +// ...later, when the object is not needed any more: +obj = null; + +// Can't get cache.size, as it's a WeakMap, +// but it's 0 or soon be 0 +// When obj gets garbage collected, cached data will be removed as well +``` + +## WeakSet + +[`WeakSet`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakSet) behaves similarly: + +- It is analogous to `Set`, but we may only add objects to `WeakSet` (not primitives). +- An object exists in the set while it is reachable from somewhere else. +- Like `Set`, it supports [`add`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Weakset/add), [`has`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Weakset/has) and [`delete`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Weakset/delete), but not `size`, `keys()` and no iterations. + +Being "weak", it also serves as additional storage. But not for arbitrary data, rather for "yes/no" facts. A membership in `WeakSet` may mean something about the object. + +For instance, we can add users to `WeakSet` to keep track of those who visited our site: + +```js run +let visitedSet = new WeakSet(); + +let john = { name: "John" }; +let pete = { name: "Pete" }; +let mary = { name: "Mary" }; + +visitedSet.add(john); // John visited us +visitedSet.add(pete); // Then Pete +visitedSet.add(john); // John again + +// visitedSet has 2 users now + +// check if John visited? +alert(visitedSet.has(john)); // true + +// check if Mary visited? +alert(visitedSet.has(mary)); // false + +john = null; + +// visitedSet will be cleaned automatically +``` + +The most notable limitation of `WeakMap` and `WeakSet` is the absence of iterations, and the inability to get all current content. That may appear inconvenient, but does not prevent `WeakMap/WeakSet` from doing their main job -- be an "additional" storage of data for objects which are stored/managed at another place. + +## Summary + +[`WeakMap`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakMap) is `Map`-like collection that allows only objects as keys and removes them together with associated value once they become inaccessible by other means. + +[`WeakSet`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WeakSet) is `Set`-like collection that stores only objects and removes them once they become inaccessible by other means. + +Their main advantages are that they have weak reference to objects, so they can easily be removed by garbage collector. + +That comes at the cost of not having support for `clear`, `size`, `keys`, `values`... + +`WeakMap` and `WeakSet` are used as "secondary" data structures in addition to the "primary" object storage. Once the object is removed from the primary storage, if it is only found as the key of `WeakMap` or in a `WeakSet`, it will be cleaned up automatically. diff --git a/1-js/05-data-types/08-keys-values-entries/01-sum-salaries/_js.view/solution.js b/1-js/05-data-types/09-keys-values-entries/01-sum-salaries/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/01-sum-salaries/_js.view/solution.js rename to 1-js/05-data-types/09-keys-values-entries/01-sum-salaries/_js.view/solution.js diff --git a/1-js/05-data-types/08-keys-values-entries/01-sum-salaries/_js.view/test.js b/1-js/05-data-types/09-keys-values-entries/01-sum-salaries/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/01-sum-salaries/_js.view/test.js rename to 1-js/05-data-types/09-keys-values-entries/01-sum-salaries/_js.view/test.js diff --git a/1-js/05-data-types/08-keys-values-entries/01-sum-salaries/solution.md b/1-js/05-data-types/09-keys-values-entries/01-sum-salaries/solution.md similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/01-sum-salaries/solution.md rename to 1-js/05-data-types/09-keys-values-entries/01-sum-salaries/solution.md diff --git a/1-js/05-data-types/08-keys-values-entries/01-sum-salaries/task.md b/1-js/05-data-types/09-keys-values-entries/01-sum-salaries/task.md similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/01-sum-salaries/task.md rename to 1-js/05-data-types/09-keys-values-entries/01-sum-salaries/task.md diff --git a/1-js/05-data-types/08-keys-values-entries/02-count-properties/_js.view/solution.js b/1-js/05-data-types/09-keys-values-entries/02-count-properties/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/02-count-properties/_js.view/solution.js rename to 1-js/05-data-types/09-keys-values-entries/02-count-properties/_js.view/solution.js diff --git a/1-js/05-data-types/08-keys-values-entries/02-count-properties/_js.view/test.js b/1-js/05-data-types/09-keys-values-entries/02-count-properties/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/02-count-properties/_js.view/test.js rename to 1-js/05-data-types/09-keys-values-entries/02-count-properties/_js.view/test.js diff --git a/1-js/05-data-types/09-destructuring-assignment/6-max-salary/solution.md b/1-js/05-data-types/09-keys-values-entries/02-count-properties/solution.md similarity index 100% rename from 1-js/05-data-types/09-destructuring-assignment/6-max-salary/solution.md rename to 1-js/05-data-types/09-keys-values-entries/02-count-properties/solution.md diff --git a/1-js/05-data-types/08-keys-values-entries/02-count-properties/task.md b/1-js/05-data-types/09-keys-values-entries/02-count-properties/task.md similarity index 100% rename from 1-js/05-data-types/08-keys-values-entries/02-count-properties/task.md rename to 1-js/05-data-types/09-keys-values-entries/02-count-properties/task.md diff --git a/1-js/05-data-types/08-keys-values-entries/article.md b/1-js/05-data-types/09-keys-values-entries/article.md similarity index 58% rename from 1-js/05-data-types/08-keys-values-entries/article.md rename to 1-js/05-data-types/09-keys-values-entries/article.md index 66ca3ca92..bef678f53 100644 --- a/1-js/05-data-types/08-keys-values-entries/article.md +++ b/1-js/05-data-types/09-keys-values-entries/article.md @@ -1,17 +1,17 @@ # Object.keys, values, entries -Let's step away from the individual data structures and talk about the iterations over them. +Let's step away from the individual data structures and talk about the iterations over them. In the previous chapter we saw methods `map.keys()`, `map.values()`, `map.entries()`. -These methods are generic, there is a common agreement to use them for data structures. If we ever create a data structure of our own, we should implement them too. +These methods are generic, there is a common agreement to use them for data structures. If we ever create a data structure of our own, we should implement them too. They are supported for: - `Map` - `Set` -- `Array` (except `arr.values()`) +- `Array` Plain objects also support similar methods, but the syntax is a bit different. @@ -23,7 +23,7 @@ For plain objects, the following methods are available: - [Object.values(obj)](mdn:js/Object/values) -- returns an array of values. - [Object.entries(obj)](mdn:js/Object/entries) -- returns an array of `[key, value]` pairs. -...But please note the distinctions (compared to map for example): +Please note the distinctions (compared to map for example): | | Map | Object | |-------------|------------------|--------------| @@ -32,7 +32,7 @@ For plain objects, the following methods are available: The first difference is that we have to call `Object.keys(obj)`, and not `obj.keys()`. -Why so? The main reason is flexibility. Remember, objects are a base of all complex structures in JavaScript. So we may have an object of our own like `order` that implements its own `order.values()` method. And we still can call `Object.values(order)` on it. +Why so? The main reason is flexibility. Remember, objects are a base of all complex structures in JavaScript. So we may have an object of our own like `data` that implements its own `data.values()` method. And we still can call `Object.values(data)` on it. The second difference is that `Object.*` methods return "real" array objects, not just an iterable. That's mainly for historical reasons. @@ -63,8 +63,41 @@ for (let value of Object.values(user)) { } ``` -## Object.keys/values/entries ignore symbolic properties - +```warn header="Object.keys/values/entries ignore symbolic properties" Just like a `for..in` loop, these methods ignore properties that use `Symbol(...)` as keys. -Usually that's convenient. But if we want symbolic keys too, then there's a separate method [Object.getOwnPropertySymbols](mdn:js/Object/getOwnPropertySymbols) that returns an array of only symbolic keys. Also, the method [Reflect.ownKeys(obj)](mdn:js/Reflect/ownKeys) returns *all* keys. +Usually that's convenient. But if we want symbolic keys too, then there's a separate method [Object.getOwnPropertySymbols](mdn:js/Object/getOwnPropertySymbols) that returns an array of only symbolic keys. Also, there exist a method [Reflect.ownKeys(obj)](mdn:js/Reflect/ownKeys) that returns *all* keys. +``` + + +## Transforming objects + +Objects lack many methods that exist for arrays, e.g. `map`, `filter` and others. + +If we'd like to apply them, then we can use `Object.entries` followed by `Object.fromEntries`: + +1. Use `Object.entries(obj)` to get an array of key/value pairs from `obj`. +2. Use array methods on that array, e.g. `map`, to transform these key/value pairs. +3. Use `Object.fromEntries(array)` on the resulting array to turn it back into an object. + +For example, we have an object with prices, and would like to double them: + +```js run +let prices = { + banana: 1, + orange: 2, + meat: 4, +}; + +*!* +let doublePrices = Object.fromEntries( + // convert prices to array, map each key/value pair into another pair + // and then fromEntries gives back the object + Object.entries(prices).map(entry => [entry[0], entry[1] * 2]) +); +*/!* + +alert(doublePrices.meat); // 8 +``` + +It may look difficult at first sight, but becomes easy to understand after you use it once or twice. We can make powerful chains of transforms this way. diff --git a/1-js/05-data-types/10-date/1-new-date/solution.md b/1-js/05-data-types/10-date/1-new-date/solution.md deleted file mode 100644 index eb271a91a..000000000 --- a/1-js/05-data-types/10-date/1-new-date/solution.md +++ /dev/null @@ -1,8 +0,0 @@ -The `new Date` constructor uses the local time zone by default. So the only important thing to remember is that months start from zero. - -So February has number 1. - -```js run -let d = new Date(2012, 1, 20, 3, 12); -alert( d ); -``` diff --git a/1-js/05-data-types/09-destructuring-assignment/1-destruct-user/solution.md b/1-js/05-data-types/10-destructuring-assignment/1-destruct-user/solution.md similarity index 100% rename from 1-js/05-data-types/09-destructuring-assignment/1-destruct-user/solution.md rename to 1-js/05-data-types/10-destructuring-assignment/1-destruct-user/solution.md diff --git a/1-js/05-data-types/09-destructuring-assignment/1-destruct-user/task.md b/1-js/05-data-types/10-destructuring-assignment/1-destruct-user/task.md similarity index 76% rename from 1-js/05-data-types/09-destructuring-assignment/1-destruct-user/task.md rename to 1-js/05-data-types/10-destructuring-assignment/1-destruct-user/task.md index b2213323a..b68db5c59 100644 --- a/1-js/05-data-types/09-destructuring-assignment/1-destruct-user/task.md +++ b/1-js/05-data-types/10-destructuring-assignment/1-destruct-user/task.md @@ -17,9 +17,9 @@ Write the destructuring assignment that reads: - `name` property into the variable `name`. - `years` property into the variable `age`. -- `isAdmin` property into the variable `isAdmin` (false if absent) +- `isAdmin` property into the variable `isAdmin` (false, if no such property) -The values after the assignment should be: +Here's an example of the values after your assignment: ```js let user = { name: "John", years: 30 }; diff --git a/1-js/05-data-types/09-destructuring-assignment/6-max-salary/_js.view/solution.js b/1-js/05-data-types/10-destructuring-assignment/6-max-salary/_js.view/solution.js similarity index 67% rename from 1-js/05-data-types/09-destructuring-assignment/6-max-salary/_js.view/solution.js rename to 1-js/05-data-types/10-destructuring-assignment/6-max-salary/_js.view/solution.js index f4bd5c761..6538af42b 100644 --- a/1-js/05-data-types/09-destructuring-assignment/6-max-salary/_js.view/solution.js +++ b/1-js/05-data-types/10-destructuring-assignment/6-max-salary/_js.view/solution.js @@ -1,16 +1,14 @@ function topSalary(salaries) { - let max = 0; + let maxSalary = 0; let maxName = null; for(const [name, salary] of Object.entries(salaries)) { - if (max < salary) { - max = salary; + if (maxSalary < salary) { + maxSalary = salary; maxName = name; } } return maxName; -} - - +} \ No newline at end of file diff --git a/1-js/05-data-types/09-destructuring-assignment/6-max-salary/_js.view/test.js b/1-js/05-data-types/10-destructuring-assignment/6-max-salary/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/09-destructuring-assignment/6-max-salary/_js.view/test.js rename to 1-js/05-data-types/10-destructuring-assignment/6-max-salary/_js.view/test.js diff --git a/1-js/05-data-types/10-date/3-weekday/solution.md b/1-js/05-data-types/10-destructuring-assignment/6-max-salary/solution.md similarity index 100% rename from 1-js/05-data-types/10-date/3-weekday/solution.md rename to 1-js/05-data-types/10-destructuring-assignment/6-max-salary/solution.md diff --git a/1-js/05-data-types/09-destructuring-assignment/6-max-salary/task.md b/1-js/05-data-types/10-destructuring-assignment/6-max-salary/task.md similarity index 100% rename from 1-js/05-data-types/09-destructuring-assignment/6-max-salary/task.md rename to 1-js/05-data-types/10-destructuring-assignment/6-max-salary/task.md diff --git a/1-js/05-data-types/09-destructuring-assignment/article.md b/1-js/05-data-types/10-destructuring-assignment/article.md similarity index 55% rename from 1-js/05-data-types/09-destructuring-assignment/article.md rename to 1-js/05-data-types/10-destructuring-assignment/article.md index 2f918e566..376d332d9 100644 --- a/1-js/05-data-types/09-destructuring-assignment/article.md +++ b/1-js/05-data-types/10-destructuring-assignment/article.md @@ -2,37 +2,48 @@ The two most used data structures in JavaScript are `Object` and `Array`. -Objects allow us to pack many pieces of information into a single entity and arrays allow us to store ordered collections. So we can make an object or an array and handle it as a single entity, or maybe pass it to a function call. +- Objects allow us to create a single entity that stores data items by key. +- Arrays allow us to gather data items into an ordered list. -*Destructuring assignment* is a special syntax that allows us to "unpack" arrays or objects into a bunch of variables, as sometimes they are more convenient. Destructuring also works great with complex functions that have a lot of parameters, default values, and soon we'll see how these are handled too. +However, when we pass these to a function, we may not need all of it. The function might only require certain elements or properties. + +*Destructuring assignment* is a special syntax that allows us to "unpack" arrays or objects into a bunch of variables, as sometimes that's more convenient. + +Destructuring also works well with complex functions that have a lot of parameters, default values, and so on. Soon we'll see that. ## Array destructuring -An example of how the array is destructured into variables: +Here's an example of how an array is destructured into variables: ```js -// we have an array with the name and surname -let arr = ["Ilya", "Kantor"] +// we have an array with a name and surname +let arr = ["John", "Smith"] *!* // destructuring assignment +// sets firstName = arr[0] +// and surname = arr[1] let [firstName, surname] = arr; */!* -alert(firstName); // Ilya -alert(surname); // Kantor +alert(firstName); // John +alert(surname); // Smith ``` Now we can work with variables instead of array members. It looks great when combined with `split` or other array-returning methods: -```js -let [firstName, surname] = "Ilya Kantor".split(' '); +```js run +let [firstName, surname] = "John Smith".split(' '); +alert(firstName); // John +alert(surname); // Smith ``` +As you can see, the syntax is simple. There are several peculiar details though. Let's see more examples to understand it better. + ````smart header="\"Destructuring\" does not mean \"destructive\"." -It's called "destructuring assignment," because it "destructurizes" by copying items into variables. But the array itself is not modified. +It's called "destructuring assignment," because it "destructurizes" by copying items into variables. However, the array itself is not modified. It's just a shorter way to write: ```js @@ -54,7 +65,7 @@ let [firstName, , title] = ["Julius", "Caesar", "Consul", "of the Roman Republic alert( title ); // Consul ``` -In the code above, the second element of the array is skipped, the third one is assigned to `title`, and the rest of the array is also skipped. +In the code above, the second element of the array is skipped, the third one is assigned to `title`, and the rest of the array items are also skipped (as there are no variables for them). ```` ````smart header="Works with any iterable on the right-side" @@ -65,29 +76,28 @@ In the code above, the second element of the array is skipped, the third one is let [a, b, c] = "abc"; // ["a", "b", "c"] let [one, two, three] = new Set([1, 2, 3]); ``` - +That works, because internally a destructuring assignment works by iterating over the right value. It's a kind of syntax sugar for calling `for..of` over the value to the right of `=` and assigning the values. ```` ````smart header="Assign to anything at the left-side" - -We can use any "assignables" at the left side. +We can use any "assignables" on the left side. For instance, an object property: ```js run let user = {}; -[user.name, user.surname] = "Ilya Kantor".split(' '); +[user.name, user.surname] = "John Smith".split(' '); -alert(user.name); // Ilya +alert(user.name); // John +alert(user.surname); // Smith ``` ```` ````smart header="Looping with .entries()" +In the previous chapter, we saw the [Object.entries(obj)](mdn:js/Object/entries) method. -In the previous chapter we saw the [Object.entries(obj)](mdn:js/Object/entries) method. - -We can use it with destructuring to loop over keys-and-values of an object: +We can use it with destructuring to loop over the keys-and-values of an object: ```js run let user = { @@ -95,7 +105,7 @@ let user = { age: 30 }; -// loop over keys-and-values +// loop over the keys-and-values *!* for (let [key, value] of Object.entries(user)) { */!* @@ -103,7 +113,7 @@ for (let [key, value] of Object.entries(user)) { } ``` -...And the same for a map: +The similar code for a `Map` is simpler, as it's iterable: ```js run let user = new Map(); @@ -111,35 +121,73 @@ user.set("name", "John"); user.set("age", "30"); *!* -for (let [key, value] of user.entries()) { +// Map iterates as [key, value] pairs, very convenient for destructuring +for (let [key, value] of user) { */!* alert(`${key}:${value}`); // name:John, then age:30 } ``` ```` + +````smart header="Swap variables trick" +There's a well-known trick for swapping values of two variables using a destructuring assignment: + +```js run +let guest = "Jane"; +let admin = "Pete"; + +// Let's swap the values: make guest=Pete, admin=Jane +*!* +[guest, admin] = [admin, guest]; +*/!* + +alert(`${guest} ${admin}`); // Pete Jane (successfully swapped!) +``` + +Here we create a temporary array of two variables and immediately destructure it in swapped order. + +We can swap more than two variables this way. +```` + ### The rest '...' -If we want not just to get first values, but also to gather all that follows -- we can add one more parameter that gets "the rest" using three dots `"..."`: +Usually, if the array is longer than the list at the left, the "extra" items are omitted. + +For example, here only two items are taken, and the rest is just ignored: ```js run -let [name1, name2, *!*...rest*/!*] = ["Julius", "Caesar", *!*"Consul", "of the Roman Republic"*/!*]; +let [name1, name2] = ["Julius", "Caesar", "Consul", "of the Roman Republic"]; alert(name1); // Julius alert(name2); // Caesar +// Further items aren't assigned anywhere +``` + +If we'd like also to gather all that follows -- we can add one more parameter that gets "the rest" using three dots `"..."`: + +```js run +let [name1, name2, *!*...rest*/!*] = ["Julius", "Caesar", *!*"Consul", "of the Roman Republic"*/!*]; *!* -// Note that type of `rest` is Array. +// rest is an array of items, starting from the 3rd one alert(rest[0]); // Consul alert(rest[1]); // of the Roman Republic alert(rest.length); // 2 */!* ``` -The value of `rest` is the array of the remaining array elements. We can use any other variable name in place of `rest`, just make sure it has three dots before it and goes last in the destructuring assignment. +The value of `rest` is the array of the remaining array elements. + +We can use any other variable name in place of `rest`, just make sure it has three dots before it and goes last in the destructuring assignment. + +```js run +let [name1, name2, *!*...titles*/!*] = ["Julius", "Caesar", "Consul", "of the Roman Republic"]; +// now titles = ["Consul", "of the Roman Republic"] +``` ### Default values -If there are fewer values in the array than variables in the assignment, there will be no error. Absent values are considered undefined: +If the array is shorter than the list of variables on the left, there will be no errors. Absent values are considered undefined: ```js run *!* @@ -164,7 +212,7 @@ alert(surname); // Anonymous (default used) Default values can be more complex expressions or even function calls. They are evaluated only if the value is not provided. -For instance, here we use the `prompt` function for two defaults. But it will run only for the missing one: +For instance, here we use the `prompt` function for two defaults: ```js run // runs only prompt for surname @@ -174,7 +222,7 @@ alert(name); // Julius (from array) alert(surname); // whatever prompt gets ``` - +Please note: the `prompt` will run only for the missing value (`surname`). ## Object destructuring @@ -183,10 +231,10 @@ The destructuring assignment also works with objects. The basic syntax is: ```js -let {var1, var2} = {var1:…, var2…} +let {var1, var2} = {var1:…, var2:…} ``` -We have an existing object at the right side, that we want to split into variables. The left side contains a "pattern" for corresponding properties. In the simple case, that's a list of variable names in `{...}`. +We should have an existing object on the right side, that we want to split into variables. The left side contains an object-like "pattern" for corresponding properties. In the simplest case, that's a list of variable names in `{...}`. For instance: @@ -206,16 +254,18 @@ alert(width); // 100 alert(height); // 200 ``` -Properties `options.title`, `options.width` and `options.height` are assigned to the corresponding variables. The order does not matter. This works too: +Properties `options.title`, `options.width` and `options.height` are assigned to the corresponding variables. + +The order does not matter. This works too: ```js -// changed the order of properties in let {...} +// changed the order in let {...} let {height, width, title} = { title: "Menu", height: 200, width: 100 } ``` The pattern on the left side may be more complex and specify the mapping between properties and variables. -If we want to assign a property to a variable with another name, for instance, `options.width` to go into the variable named `w`, then we can set it using a colon: +If we want to assign a property to a variable with another name, for instance, make `options.width` go into the variable named `w`, then we can set the variable name using a colon: ```js run let options = { @@ -258,7 +308,7 @@ alert(height); // 200 Just like with arrays or function parameters, default values can be any expressions or even function calls. They will be evaluated if the value is not provided. -The code below asks for width, but not the title. +In the code below `prompt` asks for `width`, but not for `title`: ```js run let options = { @@ -270,7 +320,7 @@ let {width = prompt("width?"), title = prompt("title?")} = options; */!* alert(title); // Menu -alert(width); // (whatever you the result of prompt is) +alert(width); // (whatever the result of prompt is) ``` We also can combine both the colon and equality: @@ -289,11 +339,26 @@ alert(w); // 100 alert(h); // 200 ``` -### The rest operator +If we have a complex object with many properties, we can extract only what we need: + +```js run +let options = { + title: "Menu", + width: 100, + height: 200 +}; + +// only extract title as a variable +let { title } = options; + +alert(title); // Menu +``` + +### The rest pattern "..." What if the object has more properties than we have variables? Can we take some and then assign the "rest" somewhere? -The specification for using the rest operator (three dots) here is almost in the standard, but most browsers do not support it yet. +We can use the rest pattern, just like we did with arrays. It's not supported by some older browsers (IE, use Babel to polyfill it), but works in modern ones. It looks like this: @@ -305,6 +370,8 @@ let options = { }; *!* +// title = property named title +// rest = object with the rest of properties let {title, ...rest} = options; */!* @@ -313,10 +380,8 @@ alert(rest.height); // 200 alert(rest.width); // 100 ``` - - -````smart header="Gotcha without `let`" -In the examples above variables were declared right before the assignment: `let {…} = {…}`. Of course, we could use existing variables too. But there's a catch. +````smart header="Gotcha if there's no `let`" +In the examples above variables were declared right in the assignment: `let {…} = {…}`. Of course, we could use existing variables too, without `let`. But there's a catch. This won't work: ```js run @@ -337,7 +402,9 @@ The problem is that JavaScript treats `{...}` in the main code flow (not inside } ``` -To show JavaScript that it's not a code block, we can wrap the whole assignment in parentheses `(...)`: +So here JavaScript assumes that we have a code block, that's why there's an error. We want destructuring instead. + +To show JavaScript that it's not a code block, we can wrap the expression in parentheses `(...)`: ```js run let title, width, height; @@ -347,14 +414,13 @@ let title, width, height; alert( title ); // Menu ``` - ```` ## Nested destructuring -If an object or an array contain other objects and arrays, we can use more complex left-side patterns to extract deeper portions. +If an object or an array contains other nested objects and arrays, we can use more complex left-side patterns to extract deeper portions. -In the code below `options` has another object in the property `size` and an array in the property `items`. The pattern at the left side of the assignment has the same structure: +In the code below `options` has another object in the property `size` and an array in the property `items`. The pattern on the left side of the assignment has the same structure to extract values from them: ```js run let options = { @@ -363,10 +429,10 @@ let options = { height: 200 }, items: ["Cake", "Donut"], - extra: true // something extra that we will not destruct + extra: true }; -// destructuring assignment on multiple lines for clarity +// destructuring assignment split in multiple lines for clarity let { size: { // put size here width, @@ -383,27 +449,24 @@ alert(item1); // Cake alert(item2); // Donut ``` -The whole `options` object except `extra` that was not mentioned, is assigned to corresponding variables. +All properties of `options` object except `extra` which is absent in the left part, are assigned to corresponding variables: +<<<<<<< HEAD:1-js/05-data-types/09-destructuring-assignment/article.md Note that `size` and `items` itself is not destructured. +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/05-data-types/10-destructuring-assignment/article.md  Finally, we have `width`, `height`, `item1`, `item2` and `title` from the default value. -That often happens with destructuring assignments. We have a complex object with many properties and want to extract only what we need. - -Even here it happens: -```js -// take size as a whole into a variable, ignore the rest -let { size } = options; -``` +Note that there are no variables for `size` and `items`, as we take their content instead. ## Smart function parameters -There are times when a function may have many parameters, most of which are optional. That's especially true for user interfaces. Imagine a function that creates a menu. It may have a width, a height, a title, items list and so on. +There are times when a function has many parameters, most of which are optional. That's especially true for user interfaces. Imagine a function that creates a menu. It may have a width, a height, a title, an item list and so on. -Here's a bad way to write such function: +Here's a bad way to write such a function: ```js function showMenu(title = "Untitled", width = 200, height = 100, items = []) { @@ -411,11 +474,12 @@ function showMenu(title = "Untitled", width = 200, height = 100, items = []) { } ``` -In real-life, the problem is how to remember the order of arguments. Usually IDEs try to help us, especially if the code is well-documented, but still... Another problem is how to call a function when most parameters are ok by default. +In real-life, the problem is how to remember the order of arguments. Usually, IDEs try to help us, especially if the code is well-documented, but still... Another problem is how to call a function when most parameters are ok by default. Like this? ```js +// undefined where default values are fine showMenu("My Menu", undefined, undefined, ["Item1", "Item2"]) ``` @@ -467,29 +531,28 @@ function showMenu({ showMenu(options); ``` -The syntax is the same as for a destructuring assignment: +The full syntax is the same as for a destructuring assignment: ```js function({ - incomingProperty: parameterName = defaultValue + incomingProperty: varName = defaultValue ... }) ``` +Then, for an object of parameters, there will be a variable `varName` for the property `incomingProperty`, with `defaultValue` by default. + Please note that such destructuring assumes that `showMenu()` does have an argument. If we want all values by default, then we should specify an empty object: ```js -showMenu({}); - +showMenu({}); // ok, all values are default showMenu(); // this would give an error ``` -We can fix this by making `{}` the default value for the whole destructuring thing: - +We can fix this by making `{}` the default value for the whole object of parameters: ```js run -// simplified parameters a bit for clarity -function showMenu(*!*{ title = "Menu", width = 100, height = 200 } = {}*/!*) { +function showMenu({ title = "Menu", width = 100, height = 200 }*!* = {}*/!*) { alert( `${title} ${width} ${height}` ); } @@ -501,19 +564,21 @@ In the code above, the whole arguments object is `{}` by default, so there's alw ## Summary - Destructuring assignment allows for instantly mapping an object or array onto many variables. -- The object syntax: +- The full object syntax: ```js - let {prop : varName = default, ...} = object + let {prop : varName = defaultValue, ...rest} = object ``` This means that property `prop` should go into the variable `varName` and, if no such property exists, then the `default` value should be used. -- The array syntax: + Object properties that have no mapping are copied to the `rest` object. + +- The full array syntax: ```js - let [item1 = default, item2, ...rest] = array + let [item1 = defaultValue, item2, ...rest] = array ``` - The first item goes to `item1`; the second goes into `item2`, all the rest makes the array `rest`. + The first item goes to `item1`; the second goes into `item2`, and all the rest makes the array `rest`. -- For more complex cases, the left side must have the same structure as the right one. +- It's possible to extract data from nested arrays/objects, for that the left side must have the same structure as the right one. diff --git a/1-js/05-data-types/09-destructuring-assignment/destructuring-complex.svg b/1-js/05-data-types/10-destructuring-assignment/destructuring-complex.svg similarity index 73% rename from 1-js/05-data-types/09-destructuring-assignment/destructuring-complex.svg rename to 1-js/05-data-types/10-destructuring-assignment/destructuring-complex.svg index c3e25b569..c2c0acf7c 100644 --- a/1-js/05-data-types/09-destructuring-assignment/destructuring-complex.svg +++ b/1-js/05-data-types/10-destructuring-assignment/destructuring-complex.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/05-data-types/09-destructuring-assignment/destructuring-complex.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="488px" height="150px" viewBox="0 0 488 150" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -56,4 +57,7 @@ <path id="Line-Copy-3" d="M253.5,102.5 L213.5,102.5 L213.5,100.5 L253.5,100.5 L253.5,94.5 L267.5,101.5 L253.5,108.5 L253.5,102.5 Z" fill="#EE6B47"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="488" height="150" viewBox="0 0 488 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="destructuring-complex.svg" fill-rule="nonzero"><g id="let-options-=-{" transform="translate(275.176 11.102)"><path id="let" fill="#1C85B5" d="M2.256 1.217H.226V.239h3.233v8.668h2.044v.991H0v-.99h2.256V1.216zm11.156 4.887c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795V4.033h-1.92v-.998h1.92V1.148l1.19-.307v2.194h3.083v.998H17.76V7.52c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025z"/><path id="options" fill="#181717" d="M36.682 6.412a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.38.263c-.493 0-.934-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.075-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.031-.739 3.447 3.447 0 011.381-.263c.492 0 .933.076 1.323.23.39.152.72.375.991.669.271.294.479.66.622 1.097.144.438.216.939.216 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.633-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.629.479.246.107.524.16.834.16.355 0 .66-.069.913-.208.252-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm8.805-.123c0 .61-.086 1.144-.257 1.6-.17.455-.407.833-.707 1.134a2.9 2.9 0 01-1.067.677c-.41.15-.854.225-1.333.225-.219 0-.436-.01-.653-.034a4.952 4.952 0 01-.66-.116v2.871h-1.189V3.035h1.06l.075 1.149c.342-.47.706-.8 1.094-.988.387-.19.806-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.544.396.745.694.2.299.35.659.451 1.08.1.422.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.99a2.586 2.586 0 00-.25-.773 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.636-.18c-.15 0-.303.023-.458.07a1.857 1.857 0 00-.482.24 3.59 3.59 0 00-.526.445 7 7 0 00-.591.687v3.329c.219.09.449.163.69.215.242.052.479.079.711.079.643 0 1.146-.218 1.51-.653.365-.435.548-1.09.548-1.966zm8.579 3.405a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795V4.033h-1.92v-.998h1.92V1.148l1.19-.307v2.194h3.083v.998h-3.083V7.52c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025zm4.505-5.783h-2.03v-.985h3.233v5.872h2.044v.991H53.88v-.99h2.256V4.02zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zM67.47 6.412a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.38.263c-.493 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.075-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.031-.739 3.447 3.447 0 011.381-.263c.493 0 .933.076 1.323.23.39.152.72.375.991.669.271.294.479.66.622 1.097.144.438.216.939.216 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.831-.16c-.355 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.629.479.246.107.524.16.834.16.355 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm2.94-3.432h1.059l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.581-.059.707 0 1.241.209 1.604.626.362.417.543 1.044.543 1.883v4.477h-1.19V5.517c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19V3.035zm13.131 4.99a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198V8.668c.401.114.8.2 1.196.26.397.059.791.089 1.183.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.365-.129.82-.194 1.367-.194.269 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.071.335.048.1.137.197.267.29.13.094.311.187.544.28.232.094.535.196.909.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.076.2.113.428.113.683z"/><path id="=" fill="#DBAF88" d="M97.98 5.482H92.1V4.478h5.88v1.004zm0 2.38H92.1V6.855h5.88v1.005z"/><path id="{" fill="#7E7C7B" d="M112.718 12.701h-.506c-.88 0-1.534-.206-1.962-.618-.428-.413-.643-1.038-.643-1.877V7.854a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.783-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.807-.499.33-.116.734-.174 1.213-.174h.506v.957h-.404c-1.057 0-1.585.513-1.585 1.538v1.593c0 1.112-.481 1.738-1.443 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="size" fill="#181717" d="M20.747 25.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm4.6-4.005h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.256V21.02zm.418-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.5 9.741h-5.53v-.854l3.875-5.01h-3.78v-.999h5.21v.93l-3.809 4.929h4.034v1.004zm7.936-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":{" fill="#7E7C7B" d="M48.856 19.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm17.678 4.78h-.506c-.88 0-1.533-.207-1.962-.62-.428-.412-.642-1.037-.642-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .604-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.137-1.036.09-.307.24-.57.447-.786.208-.216.477-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="width" fill="#181717" d="M37.064 37.035l-.998 6.863h-1.442l-.991-2.87-.198-.698-.226.738-.95 2.83h-1.402l-.99-6.863h1.161l.575 4.662.123 1.04.294-.91.998-3.083h.854l1.073 3.042.308.91.102-.965.534-4.696h1.175zm3.678.985h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V38.02zm.417-3.863c.132 0 .255.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.201.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm4.594 6.453c0-.583.08-1.101.24-1.555.159-.453.385-.836.68-1.148.293-.312.646-.55 1.059-.711a3.743 3.743 0 011.377-.243c.22 0 .434.014.646.041.212.027.42.07.626.13v-2.885h1.196v9.66h-1.066l-.041-1.3c-.333.484-.693.841-1.08 1.074a2.401 2.401 0 01-1.258.348c-.392 0-.737-.082-1.036-.246a2.104 2.104 0 01-.745-.693 3.33 3.33 0 01-.448-1.077 6 6 0 01-.15-1.395zm1.217-.075c0 .83.122 1.448.365 1.856.244.408.59.612 1.036.612.3 0 .619-.135.954-.403.335-.27.687-.668 1.056-1.197v-3.185a2.892 2.892 0 00-.65-.209 3.55 3.55 0 00-.704-.072c-.647 0-1.152.21-1.514.63-.362.419-.543 1.075-.543 1.968zm12.36 3.268a5.793 5.793 0 01-.835.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.683.095h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.615.684v4.552h-1.19V34.24h1.19v2.796l-.041 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477z"/><path id=":" fill="#7E7C7B" d="M71.948 36.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.146 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.146 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="100" fill="#A7333A" d="M90.31 43.898h-5.613v-1.107h2.297V36.27l-2.14 1.162-.437-1.012 2.844-1.497h1.052v7.868h1.997v1.107zm7.977-4.463c0 .683-.068 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.623-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.673 23.673 0 00-.038-.449l-3.78 2.81c.068.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .591-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14.001.282.004.423.002.142.01.278.023.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.035.936c-.413.221-.897.331-1.453.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.035-.937c.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.042c0-.155-.005-.31-.017-.462a23.67 23.67 0 00-.037-.448l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.378.516.149.144.317.256.506.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.252-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.095-.44.143-.949.143-1.527zm-4.088-.082c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="," fill="#7E7C7B" d="M108.213 45.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24V45.3z"/><path id="height" fill="#181717" d="M36.224 60.898h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.616.684v4.552h-1.189V51.24h1.19v2.796l-.042 1.08c.187-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.364-.164.551-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477zm7.977-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.475-1.914h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.255V55.02zm.417-3.863c.133 0 .256.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm9.885 3.849c.128.16.227.345.298.557.07.212.106.44.106.687 0 .355-.065.68-.195.974-.13.294-.314.546-.55.755a2.54 2.54 0 01-.852.49c-.33.115-.694.174-1.09.174-.287 0-.556-.031-.807-.093a2.064 2.064 0 01-.594-.229c-.087.128-.16.249-.22.363a.817.817 0 00-.088.382c0 .174.083.317.25.431.166.114.386.176.66.185l1.804.068c.342.01.657.052.947.13.29.077.537.189.745.335.207.146.369.326.485.54.116.214.174.462.174.745 0 .305-.066.595-.198.868a1.985 1.985 0 01-.612.721c-.275.208-.625.373-1.049.496-.424.123-.927.185-1.51.185-.557 0-1.03-.045-1.42-.134-.389-.089-.709-.212-.96-.369a1.49 1.49 0 01-.546-.56 1.51 1.51 0 01-.171-.715c0-.332.077-.624.232-.875.155-.25.394-.492.718-.724a1.133 1.133 0 01-.52-.476 1.322 1.322 0 01-.164-.632c0-.296.07-.567.209-.813.139-.246.304-.479.495-.698a3.57 3.57 0 01-.229-.307 1.91 1.91 0 01-.28-.7 2.536 2.536 0 01-.038-.462c0-.356.065-.68.195-.974.13-.294.312-.546.547-.756.235-.21.517-.372.848-.489.33-.116.696-.174 1.097-.174.169 0 .33.012.485.034.155.023.292.053.41.09h2.489v.97h-1.1zm-4.129 6.884c0 .323.169.56.506.707.337.148.807.222 1.408.222.379 0 .696-.034.954-.102.257-.069.465-.159.622-.27.157-.112.27-.24.338-.383.069-.144.103-.29.103-.441 0-.278-.114-.483-.342-.615-.228-.132-.576-.21-1.046-.233l-1.79-.061c-.151.1-.275.198-.373.294a1.292 1.292 0 00-.23.29 1.127 1.127 0 00-.116.294 1.309 1.309 0 00-.034.298zm.363-5.613c0 .22.036.42.109.602.073.182.175.337.308.465.132.127.288.226.468.297.18.07.38.106.598.106.237 0 .448-.04.632-.12.185-.08.34-.188.465-.324.126-.137.221-.294.287-.472.066-.178.1-.362.1-.554 0-.218-.037-.419-.11-.601a1.316 1.316 0 00-.308-.465 1.404 1.404 0 00-.468-.297c-.18-.071-.38-.106-.598-.106-.237 0-.448.04-.632.123-.185.082-.34.19-.465.324-.125.135-.221.29-.287.469-.066.177-.1.362-.1.553zm12.038 4.621h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.615.684v4.552h-1.19V51.24h1.19v2.796l-.041 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477zm7.71-.095a5.793 5.793 0 01-.833.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H71.64v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025z"/><path id=":" fill="#7E7C7B" d="M79.646 53.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085z"/><path id="200" fill="#A7333A" d="M98.082 60.898h-5.886v-1.066l2.31-2.297c.38-.374.687-.697.924-.97.237-.274.421-.523.553-.75a2.28 2.28 0 00.267-.638c.046-.2.068-.417.068-.65 0-.218-.03-.427-.089-.625a1.49 1.49 0 00-.273-.523 1.268 1.268 0 00-.478-.356 1.734 1.734 0 00-.705-.13c-.378 0-.722.085-1.032.253-.31.17-.595.388-.854.657l-.657-.786c.338-.356.726-.64 1.166-.855.44-.214.951-.321 1.535-.321.396 0 .757.06 1.083.178.326.118.607.289.844.512.237.224.42.497.55.82.13.324.196.689.196 1.094 0 .342-.046.659-.137.95a3.646 3.646 0 01-.414.872c-.184.29-.417.59-.697.899-.28.31-.612.65-.995 1.019l-1.62 1.579h4.341v1.134zm7.902-4.463c0 .683-.068 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.035-.937c.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.378.516.149.144.317.256.506.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.252-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.095-.44.143-.949.143-1.527zm-4.088-.081c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.989.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.623 1.45 2.925 2.925 0 01-1.035.936c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.069-1.308.208-1.873.139-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.306.27.392.18.726.458 1.004.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.018-.462a23.678 23.678 0 00-.037-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.316.256.506.335.189.08.404.12.646.12.31 0 .59-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}," fill="#7E7C7B" d="M15.784 68h.506c.88 0 1.534.206 1.962.619.428.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.604.028-.816.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.738.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V68zm7.759 11.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.439.24V79.3z"/><path id="items" fill="#181717" d="M17.65 89.02h-2.03v-.985h3.234v5.872h2.043v.991h-5.502v-.99h2.255V89.02zm.417-3.863c.133 0 .256.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.473 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.424-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.201 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.032 3.964V89.97c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489c-.107.203-.234.452-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.24.467-.39.786v4.41h-1.093v-6.864h.909l.055 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.191.242.287.616.287 1.122.11-.237.217-.448.321-.633a2.31 2.31 0 01.339-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.86 0 1.292.663 1.292 1.99v4.996h-1.094zm8.237-1.873a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.076.2.113.428.113.683z"/><path id=":[" fill="#7E7C7B" d="M56.554 87.912c.141 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm17.274 4.78h-3.37V85h3.37v.957h-2.235v10.787h2.235v.957z"/><path id=""Cake"" fill="#478964" d="M79.071 85.24l-.191 3.205h-1.121l-.198-3.206h1.51zm2.66 0l-.192 3.205h-1.121l-.198-3.206h1.51zm8.565 9.323a5.822 5.822 0 01-2.25.445c-1.262 0-2.231-.377-2.908-1.132-.677-.754-1.015-1.87-1.015-3.346a6.3 6.3 0 01.28-1.941 4.22 4.22 0 01.8-1.473 3.521 3.521 0 011.258-.933 4.03 4.03 0 011.654-.329c.415 0 .8.036 1.156.106.355.071.697.18 1.025.325v1.196a4.534 4.534 0 00-1.005-.406 4.348 4.348 0 00-1.135-.14c-.419 0-.798.078-1.138.235-.34.158-.628.385-.865.684a3.2 3.2 0 00-.547 1.094 5.19 5.19 0 00-.191 1.473c0 1.157.235 2.03.704 2.618.47.588 1.158.882 2.065.882.382 0 .75-.045 1.1-.133a5.13 5.13 0 001.012-.373v1.148zm6.398.335l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V88.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm9.591 3.124h-1.634l-3.199-3.677v3.677h-1.19V85.24h1.19v5.934l3.083-3.138h1.573l-3.22 3.165 3.397 3.698zm7.322-3.794c0 .17-.003.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.319-5.695l-.192 3.206h-1.12l-.2-3.206h1.512zm2.659 0l-.192 3.206h-1.12l-.199-3.206h1.51z"/><path id="," fill="#7E7C7B" d="M123.607 96.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24V96.3z"/><path id=""Donut"" fill="#478964" d="M140.65 85.24l-.192 3.205h-1.121l-.198-3.206h1.51zm2.659 0l-.192 3.205h-1.12l-.2-3.206h1.512zm8.962 5.092c0 .428-.034.839-.1 1.23a4.724 4.724 0 01-.317 1.094 3.69 3.69 0 01-.568.92 3.22 3.22 0 01-.854.707c-.338.196-.73.348-1.176.455-.447.107-.955.16-1.525.16h-1.914v-8.934h2.304c1.395 0 2.435.359 3.12 1.077.687.717 1.03 1.814 1.03 3.291zm-1.279.089c0-.634-.06-1.169-.178-1.607-.118-.437-.298-.79-.54-1.06a2.073 2.073 0 00-.909-.584c-.364-.12-.793-.18-1.285-.18h-1.046v6.856h.91c2.032 0 3.048-1.142 3.048-3.425zm8.846.991a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.031.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.076-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.323.23.39.152.72.375.991.669.272.294.479.66.623 1.097.143.438.215.939.215 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.623.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.63.479.245.107.523.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm2.94-3.432h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.145 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm13.213 6.863h-1.066l-.041-1.107a5.54 5.54 0 01-.584.591 3.021 3.021 0 01-.55.38 2.119 2.119 0 01-.551.201 2.84 2.84 0 01-.585.058c-.706 0-1.24-.207-1.6-.622-.36-.414-.54-1.04-.54-1.88v-4.484h1.19v4.389c0 1.053.397 1.579 1.19 1.579.146 0 .288-.022.427-.065.139-.043.285-.119.437-.226.153-.107.317-.251.493-.434.175-.182.372-.412.591-.69v-4.553h1.19v6.863zm7.711-.095a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.927-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.16.365.218 0 .457-.017.717-.051.26-.034.53-.088.813-.16v1.025zm4.348-9.564l-.191 3.206h-1.121l-.199-3.206h1.511zm2.66 0l-.192 3.206h-1.121l-.199-3.206h1.511z"/><path id="]," fill="#7E7C7B" d="M196.595 97.701h-3.37v-.957h2.221V85.957h-2.221V85h3.37v12.701zm3.985-1.401c.233.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.149.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="extra" fill="#181717" d="M21.11 108.104c0 .17-.003.31-.007.424a6.37 6.37 0 01-.021.322h-4.82c0 .701.197 1.24.589 1.616.392.376.957.564 1.695.564.2 0 .4-.008.601-.024.201-.016.395-.037.582-.065.186-.027.365-.058.536-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.194.835.194 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.555.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.61zm9.324 3.964h-1.579l-1.852-2.618-1.832 2.618h-1.538l2.659-3.445-2.536-3.418h1.524l1.805 2.639 1.77-2.639h1.47l-2.577 3.445 2.686 3.418zm7.041-.095a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.927-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.16.365.218 0 .457-.017.717-.051.26-.034.53-.088.813-.16v1.025zm2.448-6.768h1.086l.035 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H43.12c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.483.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm11.826 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.513-.39 1.27-.585 2.273-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.676-.037.415 0 .789.046 1.122.137.332.091.614.23.844.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.767.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.107.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217z"/><path id=":" fill="#7E7C7B" d="M56.554 104.912c.141 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="true" fill="#A7333A" d="M74.724 111.803a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H71.64v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm2.447-6.768h1.087l.034 1.265c.406-.488.805-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.616.69.367.46.537 1.144.51 2.051h-1.204c.014-.601-.074-1.038-.263-1.309-.189-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm12.933 6.863h-1.066l-.04-1.107a5.54 5.54 0 01-.585.591 3.021 3.021 0 01-.55.38 2.119 2.119 0 01-.551.201 2.84 2.84 0 01-.584.058c-.707 0-1.24-.207-1.6-.622-.36-.414-.54-1.04-.54-1.88v-4.484h1.19v4.389c0 1.053.396 1.579 1.189 1.579.146 0 .288-.022.427-.065.139-.043.285-.119.438-.226.152-.107.316-.251.492-.434.175-.182.372-.412.591-.69v-4.553h1.19v6.863zm7.978-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id="}" fill="#7E7C7B" d="M.39 119h.506c.88 0 1.533.206 1.961.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.09.307-.239.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.216.174H.39v-.957H.8c1.053 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565H.39V119z"/></g><g id="let-{-size:-{-width," transform="translate(12.176 11.102)"><path id="let" fill="#1C85B5" d="M2.256 1.217H.226V.239h3.233v8.668h2.044v.991H0v-.99h2.256V1.216zm11.156 4.887c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795V4.033h-1.92v-.998h1.92V1.148l1.19-.307v2.194h3.083v.998H17.76V7.52c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025z"/><path id="{" fill="#7E7C7B" d="M35.745 12.701h-.506c-.88 0-1.533-.206-1.962-.618-.428-.413-.642-1.038-.642-1.877V7.854a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .603-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.136-1.036.092-.307.24-.57.448-.786.208-.216.476-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.058 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.443 1.88.971.095 1.457.72 1.457 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="size" fill="#181717" d="M20.747 25.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm4.6-4.005h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.256V21.02zm.418-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.5 9.741h-5.53v-.854l3.875-5.01h-3.78v-.999h5.21v.93l-3.809 4.929h4.034v1.004zm7.936-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":{" fill="#7E7C7B" d="M48.856 19.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm17.678 4.78h-.506c-.88 0-1.533-.207-1.962-.62-.428-.412-.642-1.037-.642-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .604-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.137-1.036.09-.307.24-.57.447-.786.208-.216.477-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="width" fill="#181717" d="M37.064 37.035l-.998 6.863h-1.442l-.991-2.87-.198-.698-.226.738-.95 2.83h-1.402l-.99-6.863h1.161l.575 4.662.123 1.04.294-.91.998-3.083h.854l1.073 3.042.308.91.102-.965.534-4.696h1.175zm3.678.985h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V38.02zm.417-3.863c.132 0 .255.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.201.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm4.594 6.453c0-.583.08-1.101.24-1.555.159-.453.385-.836.68-1.148.293-.312.646-.55 1.059-.711a3.743 3.743 0 011.377-.243c.22 0 .434.014.646.041.212.027.42.07.626.13v-2.885h1.196v9.66h-1.066l-.041-1.3c-.333.484-.693.841-1.08 1.074a2.401 2.401 0 01-1.258.348c-.392 0-.737-.082-1.036-.246a2.104 2.104 0 01-.745-.693 3.33 3.33 0 01-.448-1.077 6 6 0 01-.15-1.395zm1.217-.075c0 .83.122 1.448.365 1.856.244.408.59.612 1.036.612.3 0 .619-.135.954-.403.335-.27.687-.668 1.056-1.197v-3.185a2.892 2.892 0 00-.65-.209 3.55 3.55 0 00-.704-.072c-.647 0-1.152.21-1.514.63-.362.419-.543 1.075-.543 1.968zm12.36 3.268a5.793 5.793 0 01-.835.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.683.095h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.615.684v4.552h-1.19V34.24h1.19v2.796l-.041 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477z"/><path id="," fill="#7E7C7B" d="M69.727 45.3c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24V45.3z"/><path id="height" fill="#181717" d="M36.224 60.898h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.616.684v4.552h-1.189V51.24h1.19v2.796l-.042 1.08c.187-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.364-.164.551-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477zm7.977-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.475-1.914h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.255V55.02zm.417-3.863c.133 0 .256.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm9.885 3.849c.128.16.227.345.298.557.07.212.106.44.106.687 0 .355-.065.68-.195.974-.13.294-.314.546-.55.755a2.54 2.54 0 01-.852.49c-.33.115-.694.174-1.09.174-.287 0-.556-.031-.807-.093a2.064 2.064 0 01-.594-.229c-.087.128-.16.249-.22.363a.817.817 0 00-.088.382c0 .174.083.317.25.431.166.114.386.176.66.185l1.804.068c.342.01.657.052.947.13.29.077.537.189.745.335.207.146.369.326.485.54.116.214.174.462.174.745 0 .305-.066.595-.198.868a1.985 1.985 0 01-.612.721c-.275.208-.625.373-1.049.496-.424.123-.927.185-1.51.185-.557 0-1.03-.045-1.42-.134-.389-.089-.709-.212-.96-.369a1.49 1.49 0 01-.546-.56 1.51 1.51 0 01-.171-.715c0-.332.077-.624.232-.875.155-.25.394-.492.718-.724a1.133 1.133 0 01-.52-.476 1.322 1.322 0 01-.164-.632c0-.296.07-.567.209-.813.139-.246.304-.479.495-.698a3.57 3.57 0 01-.229-.307 1.91 1.91 0 01-.28-.7 2.536 2.536 0 01-.038-.462c0-.356.065-.68.195-.974.13-.294.312-.546.547-.756.235-.21.517-.372.848-.489.33-.116.696-.174 1.097-.174.169 0 .33.012.485.034.155.023.292.053.41.09h2.489v.97h-1.1zm-4.129 6.884c0 .323.169.56.506.707.337.148.807.222 1.408.222.379 0 .696-.034.954-.102.257-.069.465-.159.622-.27.157-.112.27-.24.338-.383.069-.144.103-.29.103-.441 0-.278-.114-.483-.342-.615-.228-.132-.576-.21-1.046-.233l-1.79-.061c-.151.1-.275.198-.373.294a1.292 1.292 0 00-.23.29 1.127 1.127 0 00-.116.294 1.309 1.309 0 00-.034.298zm.363-5.613c0 .22.036.42.109.602.073.182.175.337.308.465.132.127.288.226.468.297.18.07.38.106.598.106.237 0 .448-.04.632-.12.185-.08.34-.188.465-.324.126-.137.221-.294.287-.472.066-.178.1-.362.1-.554 0-.218-.037-.419-.11-.601a1.316 1.316 0 00-.308-.465 1.404 1.404 0 00-.468-.297c-.18-.071-.38-.106-.598-.106-.237 0-.448.04-.632.123-.185.082-.34.19-.465.324-.125.135-.221.29-.287.469-.066.177-.1.362-.1.553zm12.038 4.621h-1.19v-4.381c0-.529-.099-.924-.297-1.186-.198-.262-.482-.394-.851-.394-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.615.684v4.552h-1.19V51.24h1.19v2.796l-.041 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.581-.062.684 0 1.212.209 1.586.626.374.417.56 1.044.56 1.883v4.477zm7.71-.095a5.793 5.793 0 01-.833.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H71.64v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025z"/><path id="}," fill="#7E7C7B" d="M15.784 68h.506c.88 0 1.534.206 1.962.619.428.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.604.028-.816.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.738.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V68zm7.759 11.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.439.24V79.3z"/><path id="items" fill="#181717" d="M17.65 89.02h-2.03v-.985h3.234v5.872h2.043v.991h-5.502v-.99h2.255V89.02zm.417-3.863c.133 0 .256.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.473 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.424-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.201 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.032 3.964V89.97c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489c-.107.203-.234.452-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.24.467-.39.786v4.41h-1.093v-6.864h.909l.055 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.191.242.287.616.287 1.122.11-.237.217-.448.321-.633a2.31 2.31 0 01.339-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.86 0 1.292.663 1.292 1.99v4.996h-1.094zm8.237-1.873a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.076.2.113.428.113.683z"/><path id=":[" fill="#7E7C7B" d="M56.554 87.912c.141 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm17.274 4.78h-3.37V85h3.37v.957h-2.235v10.787h2.235v.957z"/><path id="item1" fill="#181717" d="M79.229 89.02h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V89.02zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.472 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.032 3.964V89.97c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489c-.107.203-.234.452-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.524 0h-5.612v-1.107h2.297V87.27l-2.14 1.162-.437-1.012 2.844-1.497h1.052v7.868h1.996v1.107z"/><path id="," fill="#7E7C7B" d="M115.91 96.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.223-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.288-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.11.136.2.307.266.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="item2" fill="#181717" d="M133.11 89.02h-2.03v-.985h3.232v5.872h2.044v.991h-5.502v-.99h2.255V89.02zm.416-3.863c.133 0 .256.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zM144 94.803a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.424-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.648.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.032 3.964V89.97c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247 3.85 3.85 0 00-.298.485c-.11.205-.24.467-.39.786v4.41h-1.093v-6.864h.909l.055 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.191.242.287.616.287 1.122.11-.237.217-.448.321-.633a2.31 2.31 0 01.339-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.86 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6 0h-5.886v-1.066l2.31-2.297c.378-.374.686-.697.923-.97.237-.274.422-.523.554-.75a2.28 2.28 0 00.266-.638c.046-.2.069-.417.069-.65 0-.218-.03-.427-.09-.625a1.49 1.49 0 00-.273-.523 1.268 1.268 0 00-.478-.356 1.734 1.734 0 00-.704-.13c-.378 0-.723.085-1.032.253-.31.17-.595.388-.855.657l-.656-.786c.337-.356.726-.64 1.165-.855.44-.214.952-.321 1.535-.321.397 0 .758.06 1.084.178.325.118.607.289.844.512.237.224.42.497.55.82.13.324.195.689.195 1.094 0 .342-.046.659-.137.95a3.646 3.646 0 01-.413.872c-.185.29-.417.59-.698.899-.28.31-.611.65-.994 1.019l-1.62 1.579h4.34v1.134z"/><path id="]," fill="#7E7C7B" d="M173.503 97.701h-3.37v-.957h2.221V85.957h-2.221V85h3.37v12.701zm3.985-1.401c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.099.447.099.725 0 .378-.07.742-.208 1.09a2.657 2.657 0 01-.623.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="title" fill="#181717" d="M20.843 111.803a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H17.76v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025zm4.505-5.783h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.472 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.927-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.16.365.218 0 .457-.017.717-.051.26-.034.53-.088.813-.16v1.025zm4.505-8.586h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.156 4.887c0 .17-.002.31-.006.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.555.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.61z"/><path id="=" fill="#AF6E24" d="M67.19 107.482h-5.878v-1.004h5.878v1.004zm0 2.38h-5.878v-1.006h5.878v1.005z"/><path id=""Menu"" fill="#478964" d="M79.071 102.24l-.191 3.205h-1.121l-.198-3.206h1.51zm2.66 0l-.192 3.205h-1.121l-.198-3.206h1.51zm9.125 9.658h-1.189l-.178-5.57-.075-2.14-.417 1.244-1.32 3.554h-.84l-1.258-3.418-.417-1.38-.027 2.235-.157 5.475h-1.149l.438-8.934h1.442l1.203 3.363.39 1.135.376-1.135 1.264-3.363h1.484l.43 8.934zm7.226-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.137-2.899h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm13.214 6.863h-1.066l-.041-1.107a5.54 5.54 0 01-.585.591 3.021 3.021 0 01-.55.38 2.119 2.119 0 01-.55.201 2.84 2.84 0 01-.585.058c-.706 0-1.24-.207-1.6-.622-.36-.414-.54-1.04-.54-1.88v-4.484h1.19v4.389c0 1.053.397 1.579 1.19 1.579.145 0 .288-.022.427-.065.139-.043.285-.119.437-.226.153-.107.317-.251.493-.434.175-.182.372-.412.59-.69v-4.553h1.19v6.863zm4.362-9.659l-.192 3.206h-1.12l-.2-3.206h1.512zm2.659 0l-.192 3.206h-1.12l-.199-3.206h1.51z"/><path id="}" fill="#7E7C7B" d="M.39 119h.506c.88 0 1.533.206 1.961.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.09.307-.239.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.216.174H.39v-.957H.8c1.053 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565H.39V119z"/></g><path id="Line-Copy" fill="#C06334" d="M253.5 44.5l14 7-14 7v-6h-40v-2h40v-6z" transform="matrix(-1 0 0 1 482 0)"/><path id="Line-Copy-2" fill="#C06334" d="M253.5 59.5l14 7-14 7v-6h-40v-2h40v-6z" transform="matrix(-1 0 0 1 482 0)"/><path id="Line-Copy-3" fill="#C06334" d="M253.5 94.5l14 7-14 7v-6h-40v-2h40v-6z" transform="matrix(-1 0 0 1 482 0)"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/05-data-types/10-destructuring-assignment/destructuring-complex.svg diff --git a/1-js/05-data-types/11-date/1-new-date/solution.md b/1-js/05-data-types/11-date/1-new-date/solution.md new file mode 100644 index 000000000..18286c336 --- /dev/null +++ b/1-js/05-data-types/11-date/1-new-date/solution.md @@ -0,0 +1,18 @@ +The `new Date` constructor uses the local time zone. So the only important thing to remember is that months start from zero. + +So February has number 1. + +Here's an example with numbers as date components: + +```js run +//new Date(year, month, date, hour, minute, second, millisecond) +let d1 = new Date(2012, 1, 20, 3, 12); +alert( d1 ); +``` +We could also create a date from a string, like this: + +```js run +//new Date(datastring) +let d2 = new Date("2012-02-20T03:12"); +alert( d2 ); +``` diff --git a/1-js/05-data-types/10-date/1-new-date/task.md b/1-js/05-data-types/11-date/1-new-date/task.md similarity index 100% rename from 1-js/05-data-types/10-date/1-new-date/task.md rename to 1-js/05-data-types/11-date/1-new-date/task.md diff --git a/1-js/05-data-types/10-date/2-get-week-day/_js.view/solution.js b/1-js/05-data-types/11-date/2-get-week-day/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/10-date/2-get-week-day/_js.view/solution.js rename to 1-js/05-data-types/11-date/2-get-week-day/_js.view/solution.js diff --git a/1-js/05-data-types/10-date/2-get-week-day/_js.view/test.js b/1-js/05-data-types/11-date/2-get-week-day/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/10-date/2-get-week-day/_js.view/test.js rename to 1-js/05-data-types/11-date/2-get-week-day/_js.view/test.js diff --git a/1-js/05-data-types/10-date/2-get-week-day/solution.md b/1-js/05-data-types/11-date/2-get-week-day/solution.md similarity index 100% rename from 1-js/05-data-types/10-date/2-get-week-day/solution.md rename to 1-js/05-data-types/11-date/2-get-week-day/solution.md diff --git a/1-js/05-data-types/10-date/2-get-week-day/task.md b/1-js/05-data-types/11-date/2-get-week-day/task.md similarity index 100% rename from 1-js/05-data-types/10-date/2-get-week-day/task.md rename to 1-js/05-data-types/11-date/2-get-week-day/task.md diff --git a/1-js/05-data-types/10-date/3-weekday/_js.view/solution.js b/1-js/05-data-types/11-date/3-weekday/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/10-date/3-weekday/_js.view/solution.js rename to 1-js/05-data-types/11-date/3-weekday/_js.view/solution.js diff --git a/1-js/05-data-types/10-date/3-weekday/_js.view/test.js b/1-js/05-data-types/11-date/3-weekday/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/10-date/3-weekday/_js.view/test.js rename to 1-js/05-data-types/11-date/3-weekday/_js.view/test.js diff --git a/1-js/11-async/01-callbacks/01-animate-circle-callback/solution.md b/1-js/05-data-types/11-date/3-weekday/solution.md similarity index 100% rename from 1-js/11-async/01-callbacks/01-animate-circle-callback/solution.md rename to 1-js/05-data-types/11-date/3-weekday/solution.md diff --git a/1-js/05-data-types/10-date/3-weekday/task.md b/1-js/05-data-types/11-date/3-weekday/task.md similarity index 100% rename from 1-js/05-data-types/10-date/3-weekday/task.md rename to 1-js/05-data-types/11-date/3-weekday/task.md diff --git a/1-js/05-data-types/10-date/4-get-date-ago/_js.view/solution.js b/1-js/05-data-types/11-date/4-get-date-ago/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/10-date/4-get-date-ago/_js.view/solution.js rename to 1-js/05-data-types/11-date/4-get-date-ago/_js.view/solution.js diff --git a/1-js/05-data-types/10-date/4-get-date-ago/_js.view/test.js b/1-js/05-data-types/11-date/4-get-date-ago/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/10-date/4-get-date-ago/_js.view/test.js rename to 1-js/05-data-types/11-date/4-get-date-ago/_js.view/test.js diff --git a/1-js/05-data-types/10-date/4-get-date-ago/solution.md b/1-js/05-data-types/11-date/4-get-date-ago/solution.md similarity index 100% rename from 1-js/05-data-types/10-date/4-get-date-ago/solution.md rename to 1-js/05-data-types/11-date/4-get-date-ago/solution.md diff --git a/1-js/05-data-types/10-date/4-get-date-ago/task.md b/1-js/05-data-types/11-date/4-get-date-ago/task.md similarity index 92% rename from 1-js/05-data-types/10-date/4-get-date-ago/task.md rename to 1-js/05-data-types/11-date/4-get-date-ago/task.md index 40dcd926d..058d39c7e 100644 --- a/1-js/05-data-types/10-date/4-get-date-ago/task.md +++ b/1-js/05-data-types/11-date/4-get-date-ago/task.md @@ -8,7 +8,7 @@ Create a function `getDateAgo(date, days)` to return the day of month `days` ago For instance, if today is 20th, then `getDateAgo(new Date(), 1)` should be 19th and `getDateAgo(new Date(), 2)` should be 18th. -Should also work over months/years reliably: +Should work reliably for `days=365` or more: ```js let date = new Date(2015, 0, 2); diff --git a/1-js/05-data-types/10-date/5-last-day-of-month/_js.view/solution.js b/1-js/05-data-types/11-date/5-last-day-of-month/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/10-date/5-last-day-of-month/_js.view/solution.js rename to 1-js/05-data-types/11-date/5-last-day-of-month/_js.view/solution.js diff --git a/1-js/05-data-types/10-date/5-last-day-of-month/_js.view/test.js b/1-js/05-data-types/11-date/5-last-day-of-month/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/10-date/5-last-day-of-month/_js.view/test.js rename to 1-js/05-data-types/11-date/5-last-day-of-month/_js.view/test.js diff --git a/1-js/05-data-types/10-date/5-last-day-of-month/solution.md b/1-js/05-data-types/11-date/5-last-day-of-month/solution.md similarity index 100% rename from 1-js/05-data-types/10-date/5-last-day-of-month/solution.md rename to 1-js/05-data-types/11-date/5-last-day-of-month/solution.md diff --git a/1-js/05-data-types/10-date/5-last-day-of-month/task.md b/1-js/05-data-types/11-date/5-last-day-of-month/task.md similarity index 100% rename from 1-js/05-data-types/10-date/5-last-day-of-month/task.md rename to 1-js/05-data-types/11-date/5-last-day-of-month/task.md diff --git a/1-js/05-data-types/10-date/6-get-seconds-today/solution.md b/1-js/05-data-types/11-date/6-get-seconds-today/solution.md similarity index 96% rename from 1-js/05-data-types/10-date/6-get-seconds-today/solution.md rename to 1-js/05-data-types/11-date/6-get-seconds-today/solution.md index a483afe93..8f8e52b68 100644 --- a/1-js/05-data-types/10-date/6-get-seconds-today/solution.md +++ b/1-js/05-data-types/11-date/6-get-seconds-today/solution.md @@ -23,4 +23,6 @@ function getSecondsToday() { let d = new Date(); return d.getHours() * 3600 + d.getMinutes() * 60 + d.getSeconds(); } + +alert( getSecondsToday() ); ``` diff --git a/1-js/05-data-types/10-date/6-get-seconds-today/task.md b/1-js/05-data-types/11-date/6-get-seconds-today/task.md similarity index 68% rename from 1-js/05-data-types/10-date/6-get-seconds-today/task.md rename to 1-js/05-data-types/11-date/6-get-seconds-today/task.md index 3fbe13286..456790928 100644 --- a/1-js/05-data-types/10-date/6-get-seconds-today/task.md +++ b/1-js/05-data-types/11-date/6-get-seconds-today/task.md @@ -2,11 +2,11 @@ importance: 5 --- -# How many seconds has passed today? +# How many seconds have passed today? Write a function `getSecondsToday()` that returns the number of seconds from the beginning of today. -For instance, if now `10:00 am`, and there was no daylight savings shift, then: +For instance, if now were `10:00 am`, and there was no daylight savings shift, then: ```js getSecondsToday() == 36000 // (3600 * 10) diff --git a/1-js/05-data-types/10-date/7-get-seconds-to-tomorrow/solution.md b/1-js/05-data-types/11-date/7-get-seconds-to-tomorrow/solution.md similarity index 100% rename from 1-js/05-data-types/10-date/7-get-seconds-to-tomorrow/solution.md rename to 1-js/05-data-types/11-date/7-get-seconds-to-tomorrow/solution.md diff --git a/1-js/05-data-types/10-date/7-get-seconds-to-tomorrow/task.md b/1-js/05-data-types/11-date/7-get-seconds-to-tomorrow/task.md similarity index 100% rename from 1-js/05-data-types/10-date/7-get-seconds-to-tomorrow/task.md rename to 1-js/05-data-types/11-date/7-get-seconds-to-tomorrow/task.md diff --git a/1-js/05-data-types/10-date/8-format-date-relative/_js.view/solution.js b/1-js/05-data-types/11-date/8-format-date-relative/_js.view/solution.js similarity index 100% rename from 1-js/05-data-types/10-date/8-format-date-relative/_js.view/solution.js rename to 1-js/05-data-types/11-date/8-format-date-relative/_js.view/solution.js diff --git a/1-js/05-data-types/10-date/8-format-date-relative/_js.view/test.js b/1-js/05-data-types/11-date/8-format-date-relative/_js.view/test.js similarity index 100% rename from 1-js/05-data-types/10-date/8-format-date-relative/_js.view/test.js rename to 1-js/05-data-types/11-date/8-format-date-relative/_js.view/test.js diff --git a/1-js/05-data-types/10-date/8-format-date-relative/solution.md b/1-js/05-data-types/11-date/8-format-date-relative/solution.md similarity index 93% rename from 1-js/05-data-types/10-date/8-format-date-relative/solution.md rename to 1-js/05-data-types/11-date/8-format-date-relative/solution.md index 2507c840c..372485685 100644 --- a/1-js/05-data-types/10-date/8-format-date-relative/solution.md +++ b/1-js/05-data-types/11-date/8-format-date-relative/solution.md @@ -40,7 +40,7 @@ alert( formatDate(new Date(new Date - 30 * 1000)) ); // "30 sec. ago" alert( formatDate(new Date(new Date - 5 * 60 * 1000)) ); // "5 min. ago" -// yesterday's date like 31.12.2016, 20:00 +// yesterday's date like 31.12.2016 20:00 alert( formatDate(new Date(new Date - 86400 * 1000)) ); ``` @@ -62,6 +62,8 @@ function formatDate(date) { year = year.toString().slice(-2); month = month < 10 ? '0' + month : month; dayOfMonth = dayOfMonth < 10 ? '0' + dayOfMonth : dayOfMonth; + hour = hour < 10 ? '0' + hour : hour; + minutes = minutes < 10 ? '0' + minutes : minutes; if (diffSec < 1) { return 'right now'; diff --git a/1-js/05-data-types/10-date/8-format-date-relative/task.md b/1-js/05-data-types/11-date/8-format-date-relative/task.md similarity index 94% rename from 1-js/05-data-types/10-date/8-format-date-relative/task.md rename to 1-js/05-data-types/11-date/8-format-date-relative/task.md index 7b341ca2e..9651b305f 100644 --- a/1-js/05-data-types/10-date/8-format-date-relative/task.md +++ b/1-js/05-data-types/11-date/8-format-date-relative/task.md @@ -20,6 +20,6 @@ alert( formatDate(new Date(new Date - 30 * 1000)) ); // "30 sec. ago" alert( formatDate(new Date(new Date - 5 * 60 * 1000)) ); // "5 min. ago" -// yesterday's date like 31.12.2016, 20:00 +// yesterday's date like 31.12.16 20:00 alert( formatDate(new Date(new Date - 86400 * 1000)) ); ``` diff --git a/1-js/05-data-types/10-date/article.md b/1-js/05-data-types/11-date/article.md similarity index 82% rename from 1-js/05-data-types/10-date/article.md rename to 1-js/05-data-types/11-date/article.md index 8a75f1cbd..6958a3a97 100644 --- a/1-js/05-data-types/10-date/article.md +++ b/1-js/05-data-types/11-date/article.md @@ -29,18 +29,24 @@ To create a new `Date` object call `new Date()` with one of the following argume alert( Jan02_1970 ); ``` - The number of milliseconds that has passed since the beginning of 1970 is called a *timestamp*. + An integer number representing the number of milliseconds that has passed since the beginning of 1970 is called a *timestamp*. It's a lightweight numeric representation of a date. We can always create a date from a timestamp using `new Date(timestamp)` and convert the existing `Date` object to a timestamp using the `date.getTime()` method (see below). -`new Date(datestring)` -: If there is a single argument, and it's a string, then it is parsed with the `Date.parse` algorithm (see below). + Dates before 01.01.1970 have negative timestamps, e.g.: + ```js run + // 31 Dec 1969 + let Dec31_1969 = new Date(-24 * 3600 * 1000); + alert( Dec31_1969 ); + ``` +`new Date(datestring)` +: If there is a single argument, and it's a string, then it is parsed automatically. The algorithm is the same as `Date.parse` uses, we'll cover it later. ```js run let date = new Date("2017-01-26"); alert(date); - // The time portion of the date is assumed to be midnight GMT and + // The time is not set, so it's assumed to be midnight GMT and // is adjusted according to the timezone the code is run in // So the result could be // Thu Jan 26 2017 11:00:00 GMT+1100 (Australian Eastern Daylight Time) @@ -49,11 +55,9 @@ To create a new `Date` object call `new Date()` with one of the following argume ``` `new Date(year, month, date, hours, minutes, seconds, ms)` -: Create the date with the given components in the local time zone. Only two first arguments are obligatory. - - Note: +: Create the date with the given components in the local time zone. Only the first two arguments are obligatory. - - The `year` must have 4 digits: `2013` is okay, `98` is not. + - The `year` should have 4 digits. For compatibility, 2 digits are also accepted and considered `19xx`, e.g. `98` is the same as `1998` here, but always using 4 digits is strongly encouraged. - The `month` count starts with `0` (Jan), up to `11` (Dec). - The `date` parameter is actually the day of month, if absent then `1` is assumed. - If `hours/minutes/seconds/ms` is absent, they are assumed to be equal `0`. @@ -61,11 +65,11 @@ To create a new `Date` object call `new Date()` with one of the following argume For instance: ```js - new Date(2011, 0, 1, 0, 0, 0, 0); // // 1 Jan 2011, 00:00:00 + new Date(2011, 0, 1, 0, 0, 0, 0); // 1 Jan 2011, 00:00:00 new Date(2011, 0, 1); // the same, hours etc are 0 by default ``` - The minimal precision is 1 ms (1/1000 sec): + The maximal precision is 1 ms (1/1000 sec): ```js run let date = new Date(2011, 0, 1, 2, 3, 4, 567); @@ -74,7 +78,7 @@ To create a new `Date` object call `new Date()` with one of the following argume ## Access date components -There are many methods to access the year, month and so on from the `Date` object. But they can be easily remembered when categorized. +There are methods to access the year, month and so on from the `Date` object: [getFullYear()](mdn:js/Date/getFullYear) : Get the year (4 digits) @@ -120,7 +124,7 @@ Besides the given methods, there are two special ones that do not have a UTC-var : Returns the timestamp for the date -- a number of milliseconds passed from the January 1st of 1970 UTC+0. [getTimezoneOffset()](mdn:js/Date/getTimezoneOffset) -: Returns the difference between the local time zone and UTC, in minutes: +: Returns the difference between UTC and the local time zone, in minutes: ```js run // if you are in timezone UTC-1, outputs 60 @@ -133,12 +137,12 @@ Besides the given methods, there are two special ones that do not have a UTC-var The following methods allow to set date/time components: -- [`setFullYear(year [, month, date])`](mdn:js/Date/setFullYear) -- [`setMonth(month [, date])`](mdn:js/Date/setMonth) +- [`setFullYear(year, [month], [date])`](mdn:js/Date/setFullYear) +- [`setMonth(month, [date])`](mdn:js/Date/setMonth) - [`setDate(date)`](mdn:js/Date/setDate) -- [`setHours(hour [, min, sec, ms])`](mdn:js/Date/setHours) -- [`setMinutes(min [, sec, ms])`](mdn:js/Date/setMinutes) -- [`setSeconds(sec [, ms])`](mdn:js/Date/setSeconds) +- [`setHours(hour, [min], [sec], [ms])`](mdn:js/Date/setHours) +- [`setMinutes(min, [sec], [ms])`](mdn:js/Date/setMinutes) +- [`setSeconds(sec, [ms])`](mdn:js/Date/setSeconds) - [`setMilliseconds(ms)`](mdn:js/Date/setMilliseconds) - [`setTime(milliseconds)`](mdn:js/Date/setTime) (sets the whole date by milliseconds since 01.01.1970 UTC) @@ -217,21 +221,21 @@ The important side effect: dates can be subtracted, the result is their differen That can be used for time measurements: ```js run -let start = new Date(); // start counting +let start = new Date(); // start measuring time // do the job for (let i = 0; i < 100000; i++) { let doSomething = i * i * i; } -let end = new Date(); // done +let end = new Date(); // end measuring time alert( `The loop took ${end - start} ms` ); ``` ## Date.now() -If we only want to measure the difference, we don't need the `Date` object. +If we only want to measure time, we don't need the `Date` object. There's a special method `Date.now()` that returns the current timestamp. @@ -264,6 +268,8 @@ If we want a reliable benchmark of CPU-hungry function, we should be careful. For instance, let's measure two functions that calculate the difference between two dates: which one is faster? +Such performance measurements are often called "benchmarks". + ```js // we have date1 and date2, which function faster returns their difference in ms? function diffSubtract(date1, date2) { @@ -280,7 +286,7 @@ These two do exactly the same thing, but one of them uses an explicit `date.getT So, which one is faster? -The first idea may be to run them many times in a row and measure the time difference. For our case, functions are very simple, so we have to do it around 100000 times. +The first idea may be to run them many times in a row and measure the time difference. For our case, functions are very simple, so we have to do it at least 100000 times. Let's measure: @@ -310,7 +316,7 @@ Wow! Using `getTime()` is so much faster! That's because there's no type convers Okay, we have something. But that's not a good benchmark yet. -Imagine that at the time of running `bench(diffSubtract)` CPU was doing something in parallel, and it was taking resources. And by the time of running `bench(diffGetTime)` the work has finished. +Imagine that at the time of running `bench(diffSubtract)` CPU was doing something in parallel, and it was taking resources. And by the time of running `bench(diffGetTime)` that work has finished. A pretty real scenario for a modern multi-process OS. @@ -318,7 +324,7 @@ As a result, the first benchmark will have less CPU resources than the second. T **For more reliable benchmarking, the whole pack of benchmarks should be rerun multiple times.** -Here's the code example: +For example, like this: ```js run function diffSubtract(date1, date2) { @@ -342,7 +348,7 @@ let time1 = 0; let time2 = 0; *!* -// run bench(upperSlice) and bench(upperLoop) each 10 times alternating +// run bench(diffSubtract) and bench(diffGetTime) each 10 times alternating for (let i = 0; i < 10; i++) { time1 += bench(diffSubtract); time2 += bench(diffGetTime); @@ -368,9 +374,9 @@ for (let i = 0; i < 10; i++) { ``` ```warn header="Be careful doing microbenchmarking" -Modern JavaScript engines perform many optimizations. They may tweak results of "artificial tests" compared to "normal usage", especially when we benchmark something very small. So if you seriously want to understand performance, then please study how the JavaScript engine works. And then you probably won't need microbenchmarks at all. +Modern JavaScript engines perform many optimizations. They may tweak results of "artificial tests" compared to "normal usage", especially when we benchmark something very small, such as how an operator works, or a built-in function. So if you seriously want to understand performance, then please study how the JavaScript engine works. And then you probably won't need microbenchmarks at all. -The great pack of articles about V8 can be found at <http://mrale.ph>. +The great pack of articles about V8 can be found at <https://mrale.ph>. ``` ## Date.parse from a string @@ -382,7 +388,7 @@ The string format should be: `YYYY-MM-DDTHH:mm:ss.sssZ`, where: - `YYYY-MM-DD` -- is the date: year-month-day. - The character `"T"` is used as the delimiter. - `HH:mm:ss.sss` -- is the time: hours, minutes, seconds and milliseconds. -- The optional `'Z'` part denotes the time zone in the format `+-hh:mm`. A single letter `Z` that would mean UTC+0. +- The optional `'Z'` part denotes the time zone in the format `+-hh:mm`. A single letter `Z` would mean UTC+0. Shorter variants are also possible, like `YYYY-MM-DD` or `YYYY-MM` or even `YYYY`. @@ -401,7 +407,7 @@ We can instantly create a `new Date` object from the timestamp: ```js run let date = new Date( Date.parse('2012-01-26T13:51:50.417-07:00') ); -alert(date); +alert(date); ``` ## Summary @@ -415,13 +421,13 @@ alert(date); Note that unlike many other systems, timestamps in JavaScript are in milliseconds, not in seconds. -Also, sometimes we need more precise time measurements. JavaScript itself does not have a way to measure time in microseconds (1 millionth of a second), but most environments provide it. For instance, browser has [performance.now()](mdn:api/Performance/now) that gives the number of milliseconds from the start of page loading with microsecond precision (3 digits after the point): +Sometimes we need more precise time measurements. JavaScript itself does not have a way to measure time in microseconds (1 millionth of a second), but most environments provide it. For instance, browser has [performance.now()](mdn:api/Performance/now) that gives the number of milliseconds from the start of page loading with microsecond precision (3 digits after the point): ```js run alert(`Loading started ${performance.now()}ms ago`); // Something like: "Loading started 34731.26000000001ms ago" // .26 is microseconds (260 microseconds) -// more than 3 digits after the decimal point are precision errors, but only the first 3 are correct +// more than 3 digits after the decimal point are precision errors, only the first 3 are correct ``` -Node.js has `microtime` module and other ways. Technically, any device and environment allows to get more precision, it's just not in `Date`. +Node.js has `microtime` module and other ways. Technically, almost any device and environment allows to get more precision, it's just not in `Date`. diff --git a/1-js/05-data-types/11-json/1-serialize-object/solution.md b/1-js/05-data-types/12-json/1-serialize-object/solution.md similarity index 100% rename from 1-js/05-data-types/11-json/1-serialize-object/solution.md rename to 1-js/05-data-types/12-json/1-serialize-object/solution.md diff --git a/1-js/05-data-types/11-json/1-serialize-object/task.md b/1-js/05-data-types/12-json/1-serialize-object/task.md similarity index 100% rename from 1-js/05-data-types/11-json/1-serialize-object/task.md rename to 1-js/05-data-types/12-json/1-serialize-object/task.md diff --git a/1-js/05-data-types/11-json/2-serialize-event-circular/solution.md b/1-js/05-data-types/12-json/2-serialize-event-circular/solution.md similarity index 100% rename from 1-js/05-data-types/11-json/2-serialize-event-circular/solution.md rename to 1-js/05-data-types/12-json/2-serialize-event-circular/solution.md diff --git a/1-js/05-data-types/11-json/2-serialize-event-circular/task.md b/1-js/05-data-types/12-json/2-serialize-event-circular/task.md similarity index 79% rename from 1-js/05-data-types/11-json/2-serialize-event-circular/task.md rename to 1-js/05-data-types/12-json/2-serialize-event-circular/task.md index 8b3963ddf..3755a24aa 100644 --- a/1-js/05-data-types/11-json/2-serialize-event-circular/task.md +++ b/1-js/05-data-types/12-json/2-serialize-event-circular/task.md @@ -6,7 +6,7 @@ importance: 5 In simple cases of circular references, we can exclude an offending property from serialization by its name. -But sometimes there are many backreferences. And names may be used both in circular references and normal properties. +But sometimes we can't just use the name, as it may be used both in circular references and normal properties. So we can check the property by its value. Write `replacer` function to stringify everything, but remove properties that reference `meetup`: @@ -22,7 +22,7 @@ let meetup = { }; *!* -// circular references +// circular references room.occupiedBy = meetup; meetup.self = meetup; */!* @@ -39,4 +39,3 @@ alert( JSON.stringify(meetup, function replacer(key, value) { } */ ``` - diff --git a/1-js/05-data-types/11-json/article.md b/1-js/05-data-types/12-json/article.md similarity index 83% rename from 1-js/05-data-types/11-json/article.md rename to 1-js/05-data-types/12-json/article.md index 50458906a..133ffb353 100644 --- a/1-js/05-data-types/11-json/article.md +++ b/1-js/05-data-types/12-json/article.md @@ -21,13 +21,13 @@ let user = { alert(user); // {name: "John", age: 30} ``` -...But in the process of development, new properties are added, old properties are renamed and removed. Updating such `toString` every time can become a pain. We could try to loop over properties in it, but what if the object is complex and has nested objects in properties? We'd need to implement their conversion as well. And, if we're sending the object over a network, then we also need to supply the code to "read" our object on the receiving side. +...But in the process of development, new properties are added, old properties are renamed and removed. Updating such `toString` every time can become a pain. We could try to loop over properties in it, but what if the object is complex and has nested objects in properties? We'd need to implement their conversion as well. Luckily, there's no need to write the code to handle all this. The task has been solved already. ## JSON.stringify -The [JSON](http://en.wikipedia.org/wiki/JSON) (JavaScript Object Notation) is a general format to represent values and objects. It is described as in [RFC 4627](http://tools.ietf.org/html/rfc4627) standard. Initially it was made for JavaScript, but many other languages have libraries to handle it as well. So it's easy to use JSON for data exchange when the client uses JavaScript and the server is written on Ruby/PHP/Java/Whatever. +The [JSON](https://en.wikipedia.org/wiki/JSON) (JavaScript Object Notation) is a general format to represent values and objects. It is described as in [RFC 4627](https://tools.ietf.org/html/rfc4627) standard. Initially it was made for JavaScript, but many other languages have libraries to handle it as well. So it's easy to use JSON for data exchange when the client uses JavaScript and the server is written on Ruby/PHP/Java/Whatever. JavaScript provides methods: @@ -41,7 +41,7 @@ let student = { age: 30, isAdmin: false, courses: ['html', 'css', 'js'], - wife: null + spouse: null }; *!* @@ -58,7 +58,7 @@ alert(json); "age": 30, "isAdmin": false, "courses": ["html", "css", "js"], - "wife": null + "spouse": null } */ */!* @@ -66,7 +66,7 @@ alert(json); The method `JSON.stringify(student)` takes the object and converts it into a string. -The resulting `json` string is a called *JSON-encoded* or *serialized* or *stringified* or *marshalled* object. We are ready to send it over the wire or put into a plain data store. +The resulting `json` string is called a *JSON-encoded* or *serialized* or *stringified* or *marshalled* object. We are ready to send it over the wire or put into a plain data store. Please note that a JSON-encoded object has several important differences from the object literal: @@ -76,7 +76,7 @@ Please note that a JSON-encoded object has several important differences from th `JSON.stringify` can be applied to primitives as well. -Natively supported JSON types are: +JSON supports following data types: - Objects `{ ... }` - Arrays `[ ... ]` @@ -100,12 +100,12 @@ alert( JSON.stringify(true) ); // true alert( JSON.stringify([1, 2, 3]) ); // [1,2,3] ``` -JSON is data-only cross-language specification, so some JavaScript-specific object properties are skipped by `JSON.stringify`. +JSON is data-only language-independent specification, so some JavaScript-specific object properties are skipped by `JSON.stringify`. Namely: - Function properties (methods). -- Symbolic properties. +- Symbolic keys and values. - Properties that store `undefined`. ```js run @@ -213,9 +213,9 @@ alert( JSON.stringify(meetup, *!*['title', 'participants']*/!*) ); // {"title":"Conference","participants":[{},{}]} ``` -Here we are probably too strict. The property list is applied to the whole object structure. So participants are empty, because `name` is not in the list. +Here we are probably too strict. The property list is applied to the whole object structure. So the objects in `participants` are empty, because `name` is not in the list. -Let's include every property except `room.occupiedBy` that would cause the circular reference: +Let's include in the list every property except `room.occupiedBy` that would cause the circular reference: ```js run let room = { @@ -244,7 +244,7 @@ Now everything except `occupiedBy` is serialized. But the list of properties is Fortunately, we can use a function instead of an array as the `replacer`. -The function will be called for every `(key, value)` pair and should return the "replaced" value, which will be used instead of the original one. +The function will be called for every `(key, value)` pair and should return the "replaced" value, which will be used instead of the original one. Or `undefined` if the value is to be skipped. In our case, we can return `value` "as is" for everything except `occupiedBy`. To ignore `occupiedBy`, the code below returns `undefined`: @@ -262,7 +262,7 @@ let meetup = { room.occupiedBy = meetup; // room references meetup alert( JSON.stringify(meetup, function replacer(key, value) { - alert(`${key}: ${value}`); // to see what replacer gets + alert(`${key}: ${value}`); return (key == 'occupiedBy') ? undefined : value; })); @@ -276,6 +276,7 @@ name: John name: Alice place: [object Object] number: 23 +occupiedBy: [object Object] */ ``` @@ -283,16 +284,16 @@ Please note that `replacer` function gets every key/value pair including nested The first call is special. It is made using a special "wrapper object": `{"": meetup}`. In other words, the first `(key, value)` pair has an empty key, and the value is the target object as a whole. That's why the first line is `":[object Object]"` in the example above. -The idea is to provide as much power for `replacer` as possible: it has a chance to analyze and replace/skip the whole object if necessary. +The idea is to provide as much power for `replacer` as possible: it has a chance to analyze and replace/skip even the whole object if necessary. -## Formatting: spacer +## Formatting: space -The third argument of `JSON.stringify(value, replacer, spaces)` is the number of spaces to use for pretty formatting. +The third argument of `JSON.stringify(value, replacer, space)` is the number of spaces to use for pretty formatting. -Previously, all stringified objects had no indents and extra spaces. That's fine if we want to send an object over a network. The `spacer` argument is used exclusively for a nice output. +Previously, all stringified objects had no indents and extra spaces. That's fine if we want to send an object over a network. The `space` argument is used exclusively for a nice output. -Here `spacer = 2` tells JavaScript to show nested objects on multiple lines, with indentation of 2 spaces inside an object: +Here `space = 2` tells JavaScript to show nested objects on multiple lines, with indentation of 2 spaces inside an object: ```js run let user = { @@ -328,7 +329,9 @@ alert(JSON.stringify(user, null, 2)); */ ``` -The `spaces` parameter is used solely for logging and nice-output purposes. +The third argument can also be a string. In this case, the string is used for indentation instead of a number of spaces. + +The `space` parameter is used solely for logging and nice-output purposes. ## Custom "toJSON" @@ -393,7 +396,7 @@ alert( JSON.stringify(meetup) ); */ ``` -As we can see, `toJSON` is used both for the direct call `JSON.stringify(room)` and for the nested object. +As we can see, `toJSON` is used both for the direct call `JSON.stringify(room)` and when `room` is nested in another encoded object. ## JSON.parse @@ -425,14 +428,14 @@ alert( numbers[1] ); // 1 Or for nested objects: ```js run -let user = '{ "name": "John", "age": 35, "isAdmin": false, "friends": [0,1,2,3] }'; +let userData = '{ "name": "John", "age": 35, "isAdmin": false, "friends": [0,1,2,3] }'; -user = JSON.parse(user); +let user = JSON.parse(userData); alert( user.friends[1] ); // 1 ``` -The JSON may be as complex as necessary, objects and arrays can include other objects and arrays. But they must obey the format. +The JSON may be as complex as necessary, objects and arrays can include other objects and arrays. But they must obey the same JSON format. Here are typical mistakes in hand-written JSON (sometimes we have to write it for debugging purposes): @@ -448,7 +451,7 @@ let json = `{ Besides, JSON does not support comments. Adding a comment to JSON makes it invalid. -There's another format named [JSON5](http://json5.org/), which allows unquoted keys, comments etc. But this is a standalone library, not in the specification of the language. +There's another format named [JSON5](https://json5.org/), which allows unquoted keys, comments etc. But this is a standalone library, not in the specification of the language. The regular JSON is that strict not because its developers are lazy, but to allow easy, reliable and very fast implementations of the parsing algorithm. @@ -481,7 +484,7 @@ Whoops! An error! The value of `meetup.date` is a string, not a `Date` object. How could `JSON.parse` know that it should transform that string into a `Date`? -Let's pass to `JSON.parse` the reviving function that returns all values "as is", but `date` will become a `Date`: +Let's pass to `JSON.parse` the reviving function as the second argument, that returns all values "as is", but `date` will become a `Date`: ```js run let str = '{"title":"Conference","date":"2017-11-30T12:00:00.000Z"}'; diff --git a/1-js/05-data-types/11-json/json-meetup.svg b/1-js/05-data-types/12-json/json-meetup.svg similarity index 52% rename from 1-js/05-data-types/11-json/json-meetup.svg rename to 1-js/05-data-types/12-json/json-meetup.svg index 5e1929343..f4c44f183 100644 --- a/1-js/05-data-types/11-json/json-meetup.svg +++ b/1-js/05-data-types/12-json/json-meetup.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/05-data-types/11-json/json-meetup.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="188px" height="192px" viewBox="0 0 188 192" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -32,4 +33,7 @@ <ellipse id="Oval-32" stroke="#F5A623" cx="101" cy="52.5" rx="37" ry="33.5"></ellipse> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="212" height="192" viewBox="0 0 212 192"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="json-meetup.svg"><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M71 1h89v26H71z"/><text id="number:-23" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="84" y="17">number: 23</tspan></text><path id="Rectangle-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M37 80h152v26H37z"/><text id="title:-"Conference"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="50" y="96">title: "Conference"</tspan></text><path id="Rectangle-3-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M69 157h89v26H69z"/><text id="..." fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="103" y="173">...</tspan></text><text id="place" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="139" y="56">place</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M107.5 34.5v18h6l-7 14-7-14h6v-18h2z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M119.5 35.5l7 14h-6v18h-2v-18h-6l7-14z"/><text id="occupiedBy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="11" y="56">occupiedBy</tspan></text><path id="Line-Copy-2" fill="#C06334" fill-rule="nonzero" d="M114.5 113.5v18h6l-7 14-7-14h6v-18h2z"/><text id="participants" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="2" y="135">participants</tspan></text><ellipse id="Oval-32" cx="113" cy="52.5" stroke="#478964" rx="32.582" ry="29.5"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/05-data-types/12-json/json-meetup.svg diff --git a/1-js/06-advanced-functions/01-recursion/01-sum-to/solution.md b/1-js/06-advanced-functions/01-recursion/01-sum-to/solution.md index 237b9ef9e..11667f940 100644 --- a/1-js/06-advanced-functions/01-recursion/01-sum-to/solution.md +++ b/1-js/06-advanced-functions/01-recursion/01-sum-to/solution.md @@ -37,4 +37,4 @@ P.S. Naturally, the formula is the fastest solution. It uses only 3 operations f The loop variant is the second in terms of speed. In both the recursive and the loop variant we sum the same numbers. But the recursion involves nested calls and execution stack management. That also takes resources, so it's slower. -P.P.S. The standard describes a "tail call" optimization: if the recursive call is the very last one in the function (like in `sumTo` above), then the outer function will not need to resume the execution and we don't need to remember its execution context. In that case `sumTo(100000)` is countable. But if your JavaScript engine does not support it, there will be an error: maximum stack size exceeded, because there's usually a limitation on the total stack size. +P.P.S. Some engines support the "tail call" optimization: if a recursive call is the very last one in the function, with no other calculations performed, then the outer function will not need to resume the execution, so the engine doesn't need to remember its execution context. That removes the burden on memory. But if the JavaScript engine does not support tail call optimization (most of them don't), there will be an error: maximum stack size exceeded, because there's usually a limitation on the total stack size. diff --git a/1-js/06-advanced-functions/01-recursion/02-factorial/solution.md b/1-js/06-advanced-functions/01-recursion/02-factorial/solution.md index 59040a2b7..09e511db5 100644 --- a/1-js/06-advanced-functions/01-recursion/02-factorial/solution.md +++ b/1-js/06-advanced-functions/01-recursion/02-factorial/solution.md @@ -1,4 +1,4 @@ -By definition, a factorial is `n!` can be written as `n * (n-1)!`. +By definition, a factorial `n!` can be written as `n * (n-1)!`. In other words, the result of `factorial(n)` can be calculated as `n` multiplied by the result of `factorial(n-1)`. And the call for `n-1` can recursively descend lower, and lower, till `1`. diff --git a/1-js/06-advanced-functions/01-recursion/04-output-single-linked-list/solution.md b/1-js/06-advanced-functions/01-recursion/04-output-single-linked-list/solution.md index 4e9de1469..cfcbffea5 100644 --- a/1-js/06-advanced-functions/01-recursion/04-output-single-linked-list/solution.md +++ b/1-js/06-advanced-functions/01-recursion/04-output-single-linked-list/solution.md @@ -43,7 +43,7 @@ function printList(list) { } ``` -...But that would be unwise. In the future we may need to extend a function, do something else with the list. If we change `list`, then we loose such ability. +...But that would be unwise. In the future we may need to extend a function, do something else with the list. If we change `list`, then we lose such ability. Talking about good variable names, `list` here is the list itself. The first element of it. And it should remain like that. That's clear and reliable. diff --git a/1-js/06-advanced-functions/01-recursion/05-output-single-linked-list-reverse/solution.md b/1-js/06-advanced-functions/01-recursion/05-output-single-linked-list-reverse/solution.md index a9ba0baf5..0eb76ea1c 100644 --- a/1-js/06-advanced-functions/01-recursion/05-output-single-linked-list-reverse/solution.md +++ b/1-js/06-advanced-functions/01-recursion/05-output-single-linked-list-reverse/solution.md @@ -33,11 +33,11 @@ printReverseList(list); # Using a loop -The loop variant is also a little bit more complicated then the direct output. +The loop variant is also a little bit more complicated than the direct output. There is no way to get the last value in our `list`. We also can't "go back". -So what we can do is to first go through the items in the direct order and rememeber them in an array, and then output what we remembered in the reverse order: +So what we can do is to first go through the items in the direct order and remember them in an array, and then output what we remembered in the reverse order: ```js run let list = { diff --git a/1-js/06-advanced-functions/01-recursion/article.md b/1-js/06-advanced-functions/01-recursion/article.md index d78142190..5ae894474 100644 --- a/1-js/06-advanced-functions/01-recursion/article.md +++ b/1-js/06-advanced-functions/01-recursion/article.md @@ -61,7 +61,7 @@ When `pow(x, n)` is called, the execution splits into two branches: if n==1 = x / pow(x, n) = - \ + \ else = x * pow(x, n - 1) ``` @@ -85,7 +85,7 @@ So, the recursion reduces a function call to a simpler one, and then -- to even ````smart header="Recursion is usually shorter" A recursive solution is usually shorter than an iterative one. -Here we can rewrite the same using the ternary `?` operator instead of `if` to make `pow(x, n)` more terse and still very readable: +Here we can rewrite the same using the conditional operator `?` instead of `if` to make `pow(x, n)` more terse and still very readable: ```js run function pow(x, n) { @@ -96,15 +96,15 @@ function pow(x, n) { The maximal number of nested calls (including the first one) is called *recursion depth*. In our case, it will be exactly `n`. -The maximal recursion depth is limited by JavaScript engine. We can make sure about 10000, some engines allow more, but 100000 is probably out of limit for the majority of them. There are automatic optimizations that help alleviate this ("tail calls optimizations"), but they are not yet supported everywhere and work only in simple cases. +The maximal recursion depth is limited by JavaScript engine. We can rely on it being 10000, some engines allow more, but 100000 is probably out of limit for the majority of them. There are automatic optimizations that help alleviate this ("tail calls optimizations"), but they are not yet supported everywhere and work only in simple cases. That limits the application of recursion, but it still remains very wide. There are many tasks where recursive way of thinking gives simpler code, easier to maintain. -## The execution stack +## The execution context and stack Now let's examine how recursive calls work. For that we'll look under the hood of functions. -The information about a function run is stored in its *execution context*. +The information about the process of execution of a running function is stored in its *execution context*. The [execution context](https://tc39.github.io/ecma262/#sec-execution-contexts) is an internal data structure that contains details about the execution of a function: where the control flow is now, the current variables, the value of `this` (we don't use it here) and few other internal details. @@ -132,7 +132,7 @@ We can sketch it as: </li> </ul> -That's when the function starts to execute. The condition `n == 1` is false, so the flow continues into the second branch of `if`: +That's when the function starts to execute. The condition `n == 1` is falsy, so the flow continues into the second branch of `if`: ```js run function pow(x, n) { @@ -185,7 +185,13 @@ Here's the context stack when we entered the subcall `pow(2, 2)`: The new current execution context is on top (and bold), and previous remembered contexts are below. -When we finish the subcall -- it is easy to resume the previous context, because it keeps both variables and the exact place of the code where it stopped. Here in the picture we use the word "line", but of course it's more precise. +When we finish the subcall -- it is easy to resume the previous context, because it keeps both variables and the exact place of the code where it stopped. + +```smart +Here in the picture we use the word "line", as in our example there's only one subcall in line, but generally a single line of code may contain multiple subcalls, like `pow(…) + pow(…) + somethingElse(…)`. + +So it would be more precise to say that the execution resumes "immediately after the subcall". +``` ### pow(2, 1) @@ -279,7 +285,7 @@ The iterative `pow` uses a single context changing `i` and `result` in the proce **Any recursion can be rewritten as a loop. The loop variant usually can be made more effective.** -...But sometimes the rewrite is non-trivial, especially when function uses different recursive subcalls depending on conditions and merges their results or when the branching is more intricate. And the optimization may be unneeded and totally not worth the efforts. +...But sometimes the rewrite is non-trivial, especially when a function uses different recursive subcalls depending on conditions and merges their results or when the branching is more intricate. And the optimization may be unneeded and totally not worth the efforts. Recursion can give a shorter code, easier to understand and support. Optimizations are not required in every place, mostly we need a good code, that's why it's used. @@ -296,7 +302,7 @@ let company = { salary: 1000 }, { name: 'Alice', - salary: 600 + salary: 1600 }], development: { @@ -319,32 +325,32 @@ let company = { In other words, a company has departments. - A department may have an array of staff. For instance, `sales` department has 2 employees: John and Alice. -- Or a department may split into subdepartments, like `development` has two branches: `sites` and `internals`. Each of them has the own staff. +- Or a department may split into subdepartments, like `development` has two branches: `sites` and `internals`. Each of them has their own staff. - It is also possible that when a subdepartment grows, it divides into subsubdepartments (or teams). For instance, the `sites` department in the future may be split into teams for `siteA` and `siteB`. And they, potentially, can split even more. That's not on the picture, just something to have in mind. Now let's say we want a function to get the sum of all salaries. How can we do that? -An iterative approach is not easy, because the structure is not simple. The first idea may be to make a `for` loop over `company` with nested subloop over 1st level departments. But then we need more nested subloops to iterate over the staff in 2nd level departments like `sites`. ...And then another subloop inside those for 3rd level departments that might appear in the future? Should we stop on level 3 or make 4 levels of loops? If we put 3-4 nested subloops in the code to traverse a single object, it becomes rather ugly. +An iterative approach is not easy, because the structure is not simple. The first idea may be to make a `for` loop over `company` with nested subloop over 1st level departments. But then we need more nested subloops to iterate over the staff in 2nd level departments like `sites`... And then another subloop inside those for 3rd level departments that might appear in the future? If we put 3-4 nested subloops in the code to traverse a single object, it becomes rather ugly. Let's try recursion. As we can see, when our function gets a department to sum, there are two possible cases: -1. Either it's a "simple" department with an *array of people* -- then we can sum the salaries in a simple loop. -2. Or it's *an object with `N` subdepartments* -- then we can make `N` recursive calls to get the sum for each of the subdeps and combine the results. +1. Either it's a "simple" department with an *array* of people -- then we can sum the salaries in a simple loop. +2. Or it's *an object* with `N` subdepartments -- then we can make `N` recursive calls to get the sum for each of the subdeps and combine the results. -The (1) is the base of recursion, the trivial case. +The 1st case is the base of recursion, the trivial case, when we get an array. -The (2) is the recursive step. A complex task is split into subtasks for smaller departments. They may in turn split again, but sooner or later the split will finish at (1). +The 2nd case when we get an object is the recursive step. A complex task is split into subtasks for smaller departments. They may in turn split again, but sooner or later the split will finish at (1). The algorithm is probably even easier to read from the code: ```js run let company = { // the same object, compressed for brevity - sales: [{name: 'John', salary: 1000}, {name: 'Alice', salary: 600 }], + sales: [{name: 'John', salary: 1000}, {name: 'Alice', salary: 1600 }], development: { sites: [{name: 'Peter', salary: 2000}, {name: 'Alex', salary: 1800 }], internals: [{name: 'Jack', salary: 1300}] @@ -366,7 +372,7 @@ function sumSalaries(department) { } */!* -alert(sumSalaries(company)); // 6700 +alert(sumSalaries(company)); // 7700 ``` The code is short and easy to understand (hopefully?). That's the power of recursion. It also works for any level of subdepartment nesting. @@ -416,7 +422,7 @@ let arr = [obj1, obj2, obj3]; ...But there's a problem with arrays. The "delete element" and "insert element" operations are expensive. For instance, `arr.unshift(obj)` operation has to renumber all elements to make room for a new `obj`, and if the array is big, it takes time. Same with `arr.shift()`. -The only structural modifications that do not require mass-renumbering are those that operate with the end of array: `arr.push/pop`. So an array can be quite slow for big queues. +The only structural modifications that do not require mass-renumbering are those that operate with the end of array: `arr.push/pop`. So an array can be quite slow for big queues, when we have to work with the beginning. Alternatively, if we really need fast insertion/deletion, we can choose another data structure called a [linked list](https://en.wikipedia.org/wiki/Linked_list). @@ -453,9 +459,10 @@ let list = { value: 1 }; list.next = { value: 2 }; list.next.next = { value: 3 }; list.next.next.next = { value: 4 }; +list.next.next.next.next = null; ``` -Here we can even more clearer see that there are multiple objects, each one has the `value` and `next` pointing to the neighbour. The `list` variable is the first object in the chain, so following `next` pointers from it we can reach any element. +Here we can even more clearly see that there are multiple objects, each one has the `value` and `next` pointing to the neighbour. The `list` variable is the first object in the chain, so following `next` pointers from it we can reach any element. The list can be easily split into multiple parts and later joined back: @@ -506,14 +513,17 @@ Naturally, lists are not always better than arrays. Otherwise everyone would use The main drawback is that we can't easily access an element by its number. In an array that's easy: `arr[n]` is a direct reference. But in the list we need to start from the first item and go `next` `N` times to get the Nth element. -...But we don't always need such operations. For instance, when we need a queue or even a [deque](https://en.wikipedia.org/wiki/Double-ended_queue) -- the ordered structure that must allow very fast adding/removing elements from both ends. +...But we don't always need such operations. For instance, when we need a queue or even a [deque](https://en.wikipedia.org/wiki/Double-ended_queue) -- the ordered structure that must allow very fast adding/removing elements from both ends, but access to its middle is not needed. -Sometimes it's worth to add another variable named `tail` to track the last element of the list (and update it when adding/removing elements from the end). For large sets of elements the speed difference versus arrays is huge. +Lists can be enhanced: +- We can add property `prev` in addition to `next` to reference the previous element, to move back easily. +- We can also add a variable named `tail` referencing the last element of the list (and update it when adding/removing elements from the end). +- ...The data structure may vary according to our needs. ## Summary Terms: -- *Recursion* is a programming term that means a "self-calling" function. Such functions can be used to solve certain tasks in elegant ways. +- *Recursion* is a programming term that means calling a function from itself. Recursive functions can be used to solve tasks in elegant ways. When a function calls itself, that's called a *recursion step*. The *basis* of recursion is function arguments that make the task so simple that the function does not make further calls. @@ -525,7 +535,7 @@ Terms: list = { value, next -> list } ``` - Trees like HTML elements tree or the department tree from this chapter are also naturally recursive: they branch and every branch can have other branches. + Trees like HTML elements tree or the department tree from this chapter are also naturally recursive: they have branches and every branch can have other branches. Recursive functions can be used to walk them as we've seen in the `sumSalary` example. diff --git a/1-js/06-advanced-functions/01-recursion/recursive-salaries.svg b/1-js/06-advanced-functions/01-recursion/recursive-salaries.svg index 61f32dbc0..34a832cc0 100644 --- a/1-js/06-advanced-functions/01-recursion/recursive-salaries.svg +++ b/1-js/06-advanced-functions/01-recursion/recursive-salaries.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="293" height="437" viewBox="0 0 293 437"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="recursive-salaries.svg"><g id="{" fill-rule="nonzero" transform="translate(130.107 17.102)"><path fill="#7E7C7B" d="M5.024 12.701H4.52c-.88 0-1.534-.206-1.962-.618-.429-.413-.643-1.038-.643-1.877V7.854a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096H0v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499C3.635.058 4.04 0 4.52 0h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sales" fill="#181717" d="M20.815 25.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V20.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.157 4.887c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683z"/><path id=":[{" fill="#7E7C7B" d="M56.622 19.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.281-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.281-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm17.274 4.78h-3.37V17h3.37v.957h-2.235v10.787h2.235v.957zm8.101 0h-.506c-.88 0-1.533-.207-1.962-.62-.428-.412-.642-1.037-.642-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .603-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.136-1.036.092-.307.24-.57.448-.786.208-.216.476-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.058 0-1.586.513-1.586 1.538v1.593c0 1.112-.481 1.738-1.443 1.88.971.095 1.456.72 1.456 1.873v2.338c0 1.043.525 1.565 1.573 1.565h.403v.957z"/><path id="name" fill="#181717" d="M30.775 37.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V37.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V38.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 36.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'John'" fill="#478964" d="M80.5 34.24l-.198 3.205h-1.176l-.191-3.206H80.5zm9.058.724v6.207c0 .383-.057.746-.171 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.35.15-.763.225-1.237.225-.178 0-.359-.012-.544-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.597.367.94.486.345.118.685.178 1.023.178.496 0 .88-.152 1.148-.455.269-.303.403-.74.403-1.31v-5.153h-3.397v-1.053h4.635zm8.77 5.448a4.55 4.55 0 01-.225 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.99-.673 2.98 2.98 0 01-.623-1.1c-.144-.438-.215-.942-.215-1.511 0-.533.075-1.02.225-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.324.23.39.152.72.375.99.669.272.294.48.66.623 1.097.143.438.215.939.215 1.504zm-1.217.055c0-.424-.046-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.356.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.174.212.383.372.63.479.246.107.524.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.356-.81 4.09 4.09 0 00.112-.974zm8.456 3.431h-1.19v-4.381c0-.529-.098-.924-.296-1.186-.199-.262-.482-.394-.852-.394-.159 0-.308.022-.447.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.493.434c-.177.182-.382.41-.615.684v4.552h-1.19V34.24h1.19v2.796l-.04 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.58-.062.684 0 1.213.209 1.587.626.374.417.56 1.044.56 1.883v4.477zm2.181-6.863h1.06l.047 1.108c.201-.237.395-.435.582-.592.186-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.49.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm11.238-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M123.676 45.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24V45.3z"/><path id="salary" fill="#181717" d="M36.21 59.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.407-.83-.407-.16 0-.32.029-.482.086a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.81 6.81 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 53.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="1000" fill="#A7333A" d="M98.075 60.898h-5.612v-1.107h2.297V53.27l-2.14 1.162-.437-1.012 2.843-1.497h1.053v7.868h1.996v1.107zm7.978-4.463c0 .683-.069 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="},{" fill="#7E7C7B" d="M15.853 68h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V68zm7.758 11.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24V79.3zm19.9 1.401h-.506c-.88 0-1.534-.206-1.962-.618-.428-.413-.643-1.038-.643-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.783-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.807-.499.33-.116.734-.174 1.213-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M30.775 88.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V88.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V89.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 87.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'Alice'" fill="#478964" d="M80.5 85.24l-.198 3.205h-1.176l-.191-3.206H80.5zm10.691 9.658h-1.326l-.622-1.948h-3.719l-.628 1.948H83.63l2.967-8.934h1.668l2.925 8.934zm-2.296-3.035l-1.511-4.778-1.511 4.778h3.022zm5.796-5.646h-2.03v-.978h3.234v8.668h2.043v.991h-5.502v-.99h2.255v-7.691zm7.698 2.803h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V89.02zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.274 9.489c-.31.118-.628.206-.954.263a5.845 5.845 0 01-1.008.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.879-1.447-.879-2.577 0-.542.085-1.034.253-1.476.169-.442.406-.82.711-1.135.306-.315.67-.557 1.094-.728.424-.171.89-.257 1.401-.257.356 0 .689.026.998.076.31.05.607.132.89.246v1.135a3.77 3.77 0 00-.907-.339 4.248 4.248 0 00-.953-.106c-.306 0-.594.058-.865.174-.271.117-.51.284-.714.503a2.38 2.38 0 00-.486.8c-.118.314-.177.67-.177 1.066 0 .83.201 1.45.604 1.863.404.412.963.619 1.679.619a3.897 3.897 0 001.818-.438v1.108zm8.162-3.542c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm6.679-5.695l-.199 3.206h-1.175l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M131.373 96.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.149.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="salary" fill="#181717" d="M36.21 110.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 104.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="600" fill="#A7333A" d="M98.26 109.068c0 .41-.074.796-.222 1.156-.149.36-.36.673-.633.94a2.957 2.957 0 01-.988.628 3.45 3.45 0 01-1.281.23c-.492 0-.928-.08-1.306-.236a2.32 2.32 0 01-.95-.725c-.255-.326-.449-.737-.581-1.234-.132-.497-.198-1.084-.198-1.764 0-.455.03-.897.088-1.326.06-.428.158-.831.294-1.21.137-.378.32-.724.547-1.039.228-.314.512-.584.851-.81.34-.225.74-.4 1.2-.526.46-.125.994-.188 1.6-.188h.882v1.039h-.958c-.533 0-.995.064-1.387.191-.392.128-.72.308-.985.54a2.334 2.334 0 00-.608.841c-.141.328-.23.695-.267 1.1l-.027.281c.287-.169.619-.304.995-.407a4.624 4.624 0 011.22-.153c.451 0 .846.066 1.186.198.34.132.622.315.848.55.225.235.395.516.509.844.114.328.17.688.17 1.08zm-1.244.076c0-.274-.033-.521-.1-.742a1.402 1.402 0 00-.31-.56 1.42 1.42 0 00-.537-.356 2.137 2.137 0 00-.776-.127c-.173 0-.349.016-.526.048a3.742 3.742 0 00-.523.134c-.171.056-.334.121-.49.194-.154.073-.295.15-.423.233 0 .588.041 1.079.123 1.473.082.394.204.71.366.947s.361.405.598.506c.237.1.51.15.82.15.26 0 .498-.042.715-.127a1.54 1.54 0 00.56-.372c.157-.164.28-.364.37-.598.088-.235.133-.503.133-.803zm9.037-1.71c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="}]," fill="#7E7C7B" d="M15.853 119h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V119zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.222V119h3.37v12.701zm3.986-1.401c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="development" fill="#181717" d="M15.032 159.61c0-.583.08-1.101.24-1.555.159-.453.386-.836.68-1.148.294-.312.647-.55 1.06-.711a3.743 3.743 0 011.377-.243c.218 0 .434.014.646.041.212.027.42.07.625.13v-2.885h1.196v9.66H19.79l-.041-1.3c-.333.484-.693.841-1.08 1.074a2.401 2.401 0 01-1.258.348c-.392 0-.737-.082-1.036-.246a2.104 2.104 0 01-.745-.693 3.33 3.33 0 01-.447-1.077 6 6 0 01-.15-1.395zm1.217-.075c0 .83.122 1.448.366 1.856.244.408.589.612 1.035.612.301 0 .619-.135.954-.403.335-.27.687-.668 1.056-1.197v-3.185a2.892 2.892 0 00-.65-.209 3.55 3.55 0 00-.703-.071c-.647 0-1.152.21-1.515.628-.362.42-.543 1.076-.543 1.97zm12.626-.43c0 .168-.002.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.171a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm2.495-2.899h1.353l1.682 4.54.362 1.107.376-1.135 1.668-4.512h1.306l-2.693 6.863h-1.368l-2.686-6.863zm14.137 3.07c0 .168-.003.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.171a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.476-4.717h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.334 5.195a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.031.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.076-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.323.23.39.152.72.375.991.669.272.294.479.66.622 1.097.144.438.216.939.216 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.914.208-.252.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.63.479.245.107.523.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm8.805-.123c0 .61-.086 1.144-.257 1.6-.17.455-.406.833-.707 1.134a2.9 2.9 0 01-1.067.677c-.41.15-.854.225-1.333.225-.218 0-.436-.01-.652-.034a4.952 4.952 0 01-.66-.116v2.871h-1.19v-9.666h1.06l.075 1.149c.342-.47.707-.8 1.094-.988.387-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.545.396.745.694.2.299.351.659.451 1.08.1.422.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.99a2.586 2.586 0 00-.25-.773 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.635-.18c-.15 0-.304.023-.458.07a1.857 1.857 0 00-.482.24 3.59 3.59 0 00-.527.445 7 7 0 00-.591.687v3.329c.219.09.449.163.69.215.242.052.479.079.711.079.643 0 1.146-.218 1.511-.653.365-.435.547-1.09.547-1.966zm7.943 3.5v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489 30.3 30.3 0 00-.38.748v4.41H71.47v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.134-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.244-.194.379-.246.134-.053.284-.079.448-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.109-.237.216-.448.32-.633a2.31 2.31 0 01.34-.465c.12-.125.253-.22.4-.287.145-.066.314-.099.505-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.137-2.899h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.241.209 1.603.626.363.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.489.434s-.373.41-.591.684v4.552h-1.19v-6.863zm13.228 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H94.8v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025z"/><path id=":{" fill="#7E7C7B" d="M102.806 155.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.677 4.78h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sites" fill="#181717" d="M36.21 178.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm4.6-4.005h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.418-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.472 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M72.017 172.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V170h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 190.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 189.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Peter'" fill="#478964" d="M95.895 187.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.007 3.458c0 .365-.068.726-.205 1.084a2.68 2.68 0 01-.635.96 3.227 3.227 0 01-1.101.687c-.447.176-.98.263-1.6.263h-1.107v3.206h-1.217v-8.934h2.516c.442 0 .866.049 1.271.147.406.098.762.256 1.07.475.308.219.553.501.735.848.182.346.273.767.273 1.264zm-1.264.055c0-.574-.188-1.014-.564-1.32-.376-.305-.901-.457-1.576-.457h-1.244v3.677h1.135c.72 0 1.275-.157 1.664-.471.39-.315.585-.791.585-1.43zm8.907 2.351c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.201 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.68.363.934.639.253.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm7.963-3.699c0 .17-.002.31-.006.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.555.425a2.19 2.19 0 00-.375.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.616.69.367.46.537 1.144.51 2.051h-1.203c.013-.601-.074-1.038-.264-1.309-.189-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm10.958-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M146.768 198.3c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 212.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 206.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="2000" fill="#A7333A" d="M113.545 213.898h-5.886v-1.066l2.31-2.297c.379-.374.687-.697.924-.97.237-.274.421-.523.553-.75a2.28 2.28 0 00.267-.638c.045-.2.068-.417.068-.65 0-.218-.03-.427-.089-.625a1.49 1.49 0 00-.273-.523 1.268 1.268 0 00-.479-.356 1.734 1.734 0 00-.704-.13c-.378 0-.722.085-1.032.253-.31.17-.595.388-.854.657l-.657-.786c.338-.356.726-.64 1.166-.855.44-.214.951-.321 1.535-.321.396 0 .757.06 1.083.178.326.118.607.289.844.512.237.224.42.497.55.82.13.324.195.689.195 1.094 0 .342-.045.659-.136.95a3.646 3.646 0 01-.414.872c-.184.29-.417.59-.697.899-.28.31-.612.65-.995 1.019l-1.62 1.579h4.34v1.134zm7.902-4.463c0 .683-.068 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.989.04c0 .684-.069 1.307-.206 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.035.936c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.042c0-.155-.005-.31-.017-.462a23.67 23.67 0 00-.037-.448l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="},{" fill="#7E7C7B" d="M31.247 221h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V221zm7.759 11.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm19.9 1.401h-.507c-.88 0-1.533-.206-1.962-.618-.428-.413-.642-1.038-.642-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .604-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.247-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.045-.728.137-1.036.09-.307.24-.57.447-.786.208-.216.477-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="name" fill="#181717" d="M46.17 241.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 240.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Alex'" fill="#478964" d="M95.895 238.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.69 9.658h-1.325l-.622-1.948h-3.72l-.628 1.948h-1.265l2.967-8.934h1.668l2.926 8.934zm-2.296-3.035l-1.51-4.778-1.511 4.778h3.021zm5.797-5.646h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.156 4.887c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm9.324 3.964h-1.579l-1.853-2.618-1.832 2.618h-1.538l2.66-3.445-2.537-3.418h1.525l1.804 2.639 1.771-2.639h1.47l-2.577 3.445 2.686 3.418zm5.052-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 249.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 263.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 257.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1800" fill="#A7333A" d="M113.47 264.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.772-2.242c0 .379-.077.715-.232 1.009a2.13 2.13 0 01-.65.741c-.278.2-.606.353-.984.458a4.594 4.594 0 01-1.23.157c-.488 0-.918-.054-1.29-.164a2.768 2.768 0 01-.932-.458 1.93 1.93 0 01-.759-1.586c0-.546.153-1.018.458-1.415.305-.396.777-.756 1.415-1.08-.583-.296-1.01-.625-1.278-.987a2.038 2.038 0 01-.404-1.248 2.047 2.047 0 01.715-1.552c.24-.21.539-.377.899-.502.36-.126.784-.188 1.271-.188.46 0 .865.049 1.214.147.348.098.641.238.878.42s.415.404.533.663c.119.26.178.55.178.868 0 .52-.145.962-.434 1.327-.29.364-.7.683-1.234.957.264.132.51.278.738.437a3.2 3.2 0 01.595.533c.169.196.3.415.393.657.093.241.14.51.14.806zm-1.415-4.62c0-.406-.142-.708-.427-.907-.285-.198-.685-.297-1.2-.297-.51 0-.908.098-1.193.294-.285.196-.427.49-.427.882a1.175 1.175 0 00.42.916c.144.132.323.262.537.39.214.127.47.257.766.39.515-.242.898-.494 1.148-.756a1.28 1.28 0 00.376-.913zm.123 4.71c0-.17-.027-.334-.082-.493a1.367 1.367 0 00-.294-.472 2.951 2.951 0 00-.567-.465 6.682 6.682 0 00-.903-.478c-.31.15-.57.3-.782.448a2.698 2.698 0 00-.513.45c-.13.153-.222.312-.277.476a1.632 1.632 0 00-.082.52c0 .205.045.386.133.543.09.157.211.29.366.4.155.11.34.191.554.246.214.055.447.082.697.082.242 0 .47-.025.684-.075.214-.05.4-.127.557-.23.157-.102.281-.232.372-.389.092-.157.137-.345.137-.564zm9.195-2.311c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]," fill="#7E7C7B" d="M31.247 272h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V272zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V272h3.37v12.701zm3.986-1.401c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.223-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.288-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.11.136.2.307.266.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24v-.937z"/><path id="internals" fill="#181717" d="M33.113 310.02h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.417-3.863c.132 0 .255.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.201.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm4.943 2.878h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477H42.8v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm13.227 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm7.417 0h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.449-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.355.222.146.055.324.082.534.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm10.794 6.808a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M102.806 308.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V306h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.428-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 326.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 325.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Jack'" fill="#478964" d="M95.895 323.24l-.199 3.205h-1.175l-.192-3.206h1.566zm9.057.724v6.207c0 .383-.057.746-.17 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.351.15-.764.225-1.238.225-.177 0-.358-.012-.543-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.596.367.94.486.344.118.685.178 1.022.178.497 0 .88-.152 1.149-.455.268-.303.403-.74.403-1.31v-5.153h-3.398v-1.053h4.635zm7.205 8.934l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm8.75 2.872c-.31.118-.627.206-.953.263a5.845 5.845 0 01-1.009.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.878-1.447-.878-2.577 0-.542.084-1.034.253-1.476.168-.442.405-.82.71-1.135.306-.315.67-.557 1.094-.728.424-.171.891-.257 1.402-.257.355 0 .688.026.998.076.31.05.606.132.888.246v1.135a3.77 3.77 0 00-.905-.339 4.248 4.248 0 00-.954-.106c-.305 0-.594.058-.865.174-.27.117-.51.284-.714.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.201 1.45.605 1.863.403.412.962.619 1.678.619a3.897 3.897 0 001.818-.438v1.108zm8.538.252h-1.633l-3.2-3.677v3.677h-1.189v-9.659h1.19v5.934l3.082-3.138h1.573l-3.22 3.165 3.397 3.698zm5.066-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 334.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 348.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 342.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1300" fill="#A7333A" d="M113.47 349.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.601-2.713c0 .391-.08.76-.239 1.104-.16.344-.399.644-.718.902-.319.257-.72.46-1.203.608-.483.148-1.046.222-1.688.222a12.014 12.014 0 01-1.75-.123v-1.066a11.124 11.124 0 001.86.157c.446 0 .827-.04 1.144-.12.317-.08.575-.194.776-.345.2-.15.346-.332.437-.546.092-.215.137-.456.137-.725 0-.246-.055-.46-.164-.643a1.365 1.365 0 00-.461-.458 2.281 2.281 0 00-.711-.277 4.16 4.16 0 00-.906-.092h-1.019v-.977h1.033c.269 0 .513-.036.734-.106.222-.07.41-.172.568-.304.157-.133.278-.296.362-.49.084-.193.127-.413.127-.659 0-.478-.146-.827-.438-1.046-.292-.219-.72-.328-1.285-.328-.3 0-.61.03-.93.089-.319.06-.66.148-1.025.267v-1.04c.155-.054.32-.103.496-.147a7.2 7.2 0 011.049-.177c.173-.016.34-.024.499-.024.474 0 .89.051 1.25.154.36.102.662.25.903.44.242.192.424.424.547.698.123.273.185.58.185.923 0 .51-.131.938-.393 1.285-.262.346-.621.622-1.077.827.232.036.461.11.687.219.225.11.428.25.608.42.18.171.326.373.438.605.111.233.167.49.167.773zm8.074-1.75c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]}};" fill="#7E7C7B" d="M31.247 357h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V357zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V357h3.37v12.701zM15.854 374h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V374zM.458 391h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.092.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174H.458v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V391zm7.84 11.3c.233.009.459-.012.678-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm2.14-8.388c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.28-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 37)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-4" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 168)"><path id="Line" d="M.589 2.811h13.948"/><path id="Line-Copy" d="M.589 250.189h13.948"/><path id="Line-Copy-2" d="M15.508 2.811v247.378"/></g><g id="Group-5-Copy-3" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 38 9)"><path id="Line" d="M.589 4.722h13.948"/><path id="Line-Copy" d="M.589 420.278h13.948"/><path id="Line-Copy-2" d="M15.508 4.722v415.556"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 189)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 315)"><path id="Line" d="M.589.844h13.948"/><path id="Line-Copy" d="M.589 75.156h13.948"/><path id="Line-Copy-2" d="M15.508.844v74.312"/></g><path id="Line-Copy-7" fill="#C06334" fill-rule="nonzero" d="M71.5 213.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 213.5z"/><path id="Line-Copy-8" fill="#C06334" fill-rule="nonzero" d="M129.5 323.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 323.5z"/><path id="Line-Copy-9" fill="#C06334" fill-rule="nonzero" d="M129.5 239.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 239.5z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M71.5 110.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 110.5z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="293" height="437" viewBox="0 0 293 437"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="recursive-salaries.svg"><g id="{" fill-rule="nonzero" transform="translate(130.107 17.102)"><path fill="#7E7C7B" d="M5.024 12.701H4.52c-.88 0-1.534-.206-1.962-.618-.429-.413-.643-1.038-.643-1.877V7.854a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096H0v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499C3.635.058 4.04 0 4.52 0h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sales" fill="#181717" d="M20.815 25.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V20.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.157 4.887c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683z"/><path id=":[{" fill="#7E7C7B" d="M56.622 19.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.281-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.281-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm17.274 4.78h-3.37V17h3.37v.957h-2.235v10.787h2.235v.957zm8.101 0h-.506c-.88 0-1.533-.207-1.962-.62-.428-.412-.642-1.037-.642-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .603-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.136-1.036.092-.307.24-.57.448-.786.208-.216.476-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.058 0-1.586.513-1.586 1.538v1.593c0 1.112-.481 1.738-1.443 1.88.971.095 1.456.72 1.456 1.873v2.338c0 1.043.525 1.565 1.573 1.565h.403v.957z"/><path id="name" fill="#181717" d="M30.775 37.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V37.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V38.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 36.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'John'" fill="#478964" d="M80.5 34.24l-.198 3.205h-1.176l-.191-3.206H80.5zm9.058.724v6.207c0 .383-.057.746-.171 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.35.15-.763.225-1.237.225-.178 0-.359-.012-.544-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.597.367.94.486.345.118.685.178 1.023.178.496 0 .88-.152 1.148-.455.269-.303.403-.74.403-1.31v-5.153h-3.397v-1.053h4.635zm8.77 5.448a4.55 4.55 0 01-.225 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.99-.673 2.98 2.98 0 01-.623-1.1c-.144-.438-.215-.942-.215-1.511 0-.533.075-1.02.225-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.324.23.39.152.72.375.99.669.272.294.48.66.623 1.097.143.438.215.939.215 1.504zm-1.217.055c0-.424-.046-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.356.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.174.212.383.372.63.479.246.107.524.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.356-.81 4.09 4.09 0 00.112-.974zm8.456 3.431h-1.19v-4.381c0-.529-.098-.924-.296-1.186-.199-.262-.482-.394-.852-.394-.159 0-.308.022-.447.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.493.434c-.177.182-.382.41-.615.684v4.552h-1.19V34.24h1.19v2.796l-.04 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.58-.062.684 0 1.213.209 1.587.626.374.417.56 1.044.56 1.883v4.477zm2.181-6.863h1.06l.047 1.108c.201-.237.395-.435.582-.592.186-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.49.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm11.238-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M123.676 45.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24V45.3z"/><path id="salary" fill="#181717" d="M36.21 59.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.407-.83-.407-.16 0-.32.029-.482.086a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.81 6.81 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 53.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="1000" fill="#A7333A" d="M98.075 60.898h-5.612v-1.107h2.297V53.27l-2.14 1.162-.437-1.012 2.843-1.497h1.053v7.868h1.996v1.107zm7.978-4.463c0 .683-.069 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="},{" fill="#7E7C7B" d="M15.853 68h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V68zm7.758 11.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24V79.3zm19.9 1.401h-.506c-.88 0-1.534-.206-1.962-.618-.428-.413-.643-1.038-.643-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.783-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.807-.499.33-.116.734-.174 1.213-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M30.775 88.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V88.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V89.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 87.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'Alice'" fill="#478964" d="M80.5 85.24l-.198 3.205h-1.176l-.191-3.206H80.5zm10.691 9.658h-1.326l-.622-1.948h-3.719l-.628 1.948H83.63l2.967-8.934h1.668l2.925 8.934zm-2.296-3.035l-1.511-4.778-1.511 4.778h3.022zm5.796-5.646h-2.03v-.978h3.234v8.668h2.043v.991h-5.502v-.99h2.255v-7.691zm7.698 2.803h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V89.02zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.274 9.489c-.31.118-.628.206-.954.263a5.845 5.845 0 01-1.008.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.879-1.447-.879-2.577 0-.542.085-1.034.253-1.476.169-.442.406-.82.711-1.135.306-.315.67-.557 1.094-.728.424-.171.89-.257 1.401-.257.356 0 .689.026.998.076.31.05.607.132.89.246v1.135a3.77 3.77 0 00-.907-.339 4.248 4.248 0 00-.953-.106c-.306 0-.594.058-.865.174-.271.117-.51.284-.714.503a2.38 2.38 0 00-.486.8c-.118.314-.177.67-.177 1.066 0 .83.201 1.45.604 1.863.404.412.963.619 1.679.619a3.897 3.897 0 001.818-.438v1.108zm8.162-3.542c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm6.679-5.695l-.199 3.206h-1.175l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M131.373 96.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.149.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="salary" fill="#181717" d="M36.21 110.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 104.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="600" fill="#A7333A" d="M98.26 109.068c0 .41-.074.796-.222 1.156-.149.36-.36.673-.633.94a2.957 2.957 0 01-.988.628 3.45 3.45 0 01-1.281.23c-.492 0-.928-.08-1.306-.236a2.32 2.32 0 01-.95-.725c-.255-.326-.449-.737-.581-1.234-.132-.497-.198-1.084-.198-1.764 0-.455.03-.897.088-1.326.06-.428.158-.831.294-1.21.137-.378.32-.724.547-1.039.228-.314.512-.584.851-.81.34-.225.74-.4 1.2-.526.46-.125.994-.188 1.6-.188h.882v1.039h-.958c-.533 0-.995.064-1.387.191-.392.128-.72.308-.985.54a2.334 2.334 0 00-.608.841c-.141.328-.23.695-.267 1.1l-.027.281c.287-.169.619-.304.995-.407a4.624 4.624 0 011.22-.153c.451 0 .846.066 1.186.198.34.132.622.315.848.55.225.235.395.516.509.844.114.328.17.688.17 1.08zm-1.244.076c0-.274-.033-.521-.1-.742a1.402 1.402 0 00-.31-.56 1.42 1.42 0 00-.537-.356 2.137 2.137 0 00-.776-.127c-.173 0-.349.016-.526.048a3.742 3.742 0 00-.523.134c-.171.056-.334.121-.49.194-.154.073-.295.15-.423.233 0 .588.041 1.079.123 1.473.082.394.204.71.366.947s.361.405.598.506c.237.1.51.15.82.15.26 0 .498-.042.715-.127a1.54 1.54 0 00.56-.372c.157-.164.28-.364.37-.598.088-.235.133-.503.133-.803zm9.037-1.71c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="}]," fill="#7E7C7B" d="M15.853 119h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V119zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.222V119h3.37v12.701zm3.986-1.401c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="development" fill="#181717" d="M15.032 159.61c0-.583.08-1.101.24-1.555.159-.453.386-.836.68-1.148.294-.312.647-.55 1.06-.711a3.743 3.743 0 011.377-.243c.218 0 .434.014.646.041.212.027.42.07.625.13v-2.885h1.196v9.66H19.79l-.041-1.3c-.333.484-.693.841-1.08 1.074a2.401 2.401 0 01-1.258.348c-.392 0-.737-.082-1.036-.246a2.104 2.104 0 01-.745-.693 3.33 3.33 0 01-.447-1.077 6 6 0 01-.15-1.395zm1.217-.075c0 .83.122 1.448.366 1.856.244.408.589.612 1.035.612.301 0 .619-.135.954-.403.335-.27.687-.668 1.056-1.197v-3.185a2.892 2.892 0 00-.65-.209 3.55 3.55 0 00-.703-.071c-.647 0-1.152.21-1.515.628-.362.42-.543 1.076-.543 1.97zm12.626-.43c0 .168-.002.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.171a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm2.495-2.899h1.353l1.682 4.54.362 1.107.376-1.135 1.668-4.512h1.306l-2.693 6.863h-1.368l-2.686-6.863zm14.137 3.07c0 .168-.003.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.171a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.476-4.717h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.334 5.195a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.031.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.076-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.323.23.39.152.72.375.991.669.272.294.479.66.622 1.097.144.438.216.939.216 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.914.208-.252.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.63.479.245.107.523.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm8.805-.123c0 .61-.086 1.144-.257 1.6-.17.455-.406.833-.707 1.134a2.9 2.9 0 01-1.067.677c-.41.15-.854.225-1.333.225-.218 0-.436-.01-.652-.034a4.952 4.952 0 01-.66-.116v2.871h-1.19v-9.666h1.06l.075 1.149c.342-.47.707-.8 1.094-.988.387-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.545.396.745.694.2.299.351.659.451 1.08.1.422.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.99a2.586 2.586 0 00-.25-.773 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.635-.18c-.15 0-.304.023-.458.07a1.857 1.857 0 00-.482.24 3.59 3.59 0 00-.527.445 7 7 0 00-.591.687v3.329c.219.09.449.163.69.215.242.052.479.079.711.079.643 0 1.146-.218 1.511-.653.365-.435.547-1.09.547-1.966zm7.943 3.5v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489 30.3 30.3 0 00-.38.748v4.41H71.47v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.134-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.244-.194.379-.246.134-.053.284-.079.448-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.109-.237.216-.448.32-.633a2.31 2.31 0 01.34-.465c.12-.125.253-.22.4-.287.145-.066.314-.099.505-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.137-2.899h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.241.209 1.603.626.363.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.489.434s-.373.41-.591.684v4.552h-1.19v-6.863zm13.228 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H94.8v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025z"/><path id=":{" fill="#7E7C7B" d="M102.806 155.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.677 4.78h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sites" fill="#181717" d="M36.21 178.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm4.6-4.005h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.418-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.472 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M72.017 172.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V170h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 190.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 189.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Peter'" fill="#478964" d="M95.895 187.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.007 3.458c0 .365-.068.726-.205 1.084a2.68 2.68 0 01-.635.96 3.227 3.227 0 01-1.101.687c-.447.176-.98.263-1.6.263h-1.107v3.206h-1.217v-8.934h2.516c.442 0 .866.049 1.271.147.406.098.762.256 1.07.475.308.219.553.501.735.848.182.346.273.767.273 1.264zm-1.264.055c0-.574-.188-1.014-.564-1.32-.376-.305-.901-.457-1.576-.457h-1.244v3.677h1.135c.72 0 1.275-.157 1.664-.471.39-.315.585-.791.585-1.43zm8.907 2.351c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.201 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.68.363.934.639.253.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm7.963-3.699c0 .17-.002.31-.006.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.555.425a2.19 2.19 0 00-.375.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.616.69.367.46.537 1.144.51 2.051h-1.203c.013-.601-.074-1.038-.264-1.309-.189-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm10.958-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M146.768 198.3c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 212.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 206.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="2000" fill="#A7333A" d="M113.545 213.898h-5.886v-1.066l2.31-2.297c.379-.374.687-.697.924-.97.237-.274.421-.523.553-.75a2.28 2.28 0 00.267-.638c.045-.2.068-.417.068-.65 0-.218-.03-.427-.089-.625a1.49 1.49 0 00-.273-.523 1.268 1.268 0 00-.479-.356 1.734 1.734 0 00-.704-.13c-.378 0-.722.085-1.032.253-.31.17-.595.388-.854.657l-.657-.786c.338-.356.726-.64 1.166-.855.44-.214.951-.321 1.535-.321.396 0 .757.06 1.083.178.326.118.607.289.844.512.237.224.42.497.55.82.13.324.195.689.195 1.094 0 .342-.045.659-.136.95a3.646 3.646 0 01-.414.872c-.184.29-.417.59-.697.899-.28.31-.612.65-.995 1.019l-1.62 1.579h4.34v1.134zm7.902-4.463c0 .683-.068 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.989.04c0 .684-.069 1.307-.206 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.035.936c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.042c0-.155-.005-.31-.017-.462a23.67 23.67 0 00-.037-.448l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="},{" fill="#7E7C7B" d="M31.247 221h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V221zm7.759 11.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm19.9 1.401h-.507c-.88 0-1.533-.206-1.962-.618-.428-.413-.642-1.038-.642-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .604-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.247-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.045-.728.137-1.036.09-.307.24-.57.447-.786.208-.216.477-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="name" fill="#181717" d="M46.17 241.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 240.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Alex'" fill="#478964" d="M95.895 238.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.69 9.658h-1.325l-.622-1.948h-3.72l-.628 1.948h-1.265l2.967-8.934h1.668l2.926 8.934zm-2.296-3.035l-1.51-4.778-1.511 4.778h3.021zm5.797-5.646h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.156 4.887c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm9.324 3.964h-1.579l-1.853-2.618-1.832 2.618h-1.538l2.66-3.445-2.537-3.418h1.525l1.804 2.639 1.771-2.639h1.47l-2.577 3.445 2.686 3.418zm5.052-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 249.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 263.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 257.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1800" fill="#A7333A" d="M113.47 264.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.772-2.242c0 .379-.077.715-.232 1.009a2.13 2.13 0 01-.65.741c-.278.2-.606.353-.984.458a4.594 4.594 0 01-1.23.157c-.488 0-.918-.054-1.29-.164a2.768 2.768 0 01-.932-.458 1.93 1.93 0 01-.759-1.586c0-.546.153-1.018.458-1.415.305-.396.777-.756 1.415-1.08-.583-.296-1.01-.625-1.278-.987a2.038 2.038 0 01-.404-1.248 2.047 2.047 0 01.715-1.552c.24-.21.539-.377.899-.502.36-.126.784-.188 1.271-.188.46 0 .865.049 1.214.147.348.098.641.238.878.42s.415.404.533.663c.119.26.178.55.178.868 0 .52-.145.962-.434 1.327-.29.364-.7.683-1.234.957.264.132.51.278.738.437a3.2 3.2 0 01.595.533c.169.196.3.415.393.657.093.241.14.51.14.806zm-1.415-4.62c0-.406-.142-.708-.427-.907-.285-.198-.685-.297-1.2-.297-.51 0-.908.098-1.193.294-.285.196-.427.49-.427.882a1.175 1.175 0 00.42.916c.144.132.323.262.537.39.214.127.47.257.766.39.515-.242.898-.494 1.148-.756a1.28 1.28 0 00.376-.913zm.123 4.71c0-.17-.027-.334-.082-.493a1.367 1.367 0 00-.294-.472 2.951 2.951 0 00-.567-.465 6.682 6.682 0 00-.903-.478c-.31.15-.57.3-.782.448a2.698 2.698 0 00-.513.45c-.13.153-.222.312-.277.476a1.632 1.632 0 00-.082.52c0 .205.045.386.133.543.09.157.211.29.366.4.155.11.34.191.554.246.214.055.447.082.697.082.242 0 .47-.025.684-.075.214-.05.4-.127.557-.23.157-.102.281-.232.372-.389.092-.157.137-.345.137-.564zm9.195-2.311c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]," fill="#7E7C7B" d="M31.247 272h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V272zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V272h3.37v12.701zm3.986-1.401c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.223-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.288-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.11.136.2.307.266.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24v-.937z"/><path id="internals" fill="#181717" d="M33.113 310.02h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.417-3.863c.132 0 .255.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.201.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm4.943 2.878h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477H42.8v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm13.227 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm7.417 0h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.449-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.355.222.146.055.324.082.534.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm10.794 6.808a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M102.806 308.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V306h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.428-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 326.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 325.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Jack'" fill="#478964" d="M95.895 323.24l-.199 3.205h-1.175l-.192-3.206h1.566zm9.057.724v6.207c0 .383-.057.746-.17 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.351.15-.764.225-1.238.225-.177 0-.358-.012-.543-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.596.367.94.486.344.118.685.178 1.022.178.497 0 .88-.152 1.149-.455.268-.303.403-.74.403-1.31v-5.153h-3.398v-1.053h4.635zm7.205 8.934l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm8.75 2.872c-.31.118-.627.206-.953.263a5.845 5.845 0 01-1.009.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.878-1.447-.878-2.577 0-.542.084-1.034.253-1.476.168-.442.405-.82.71-1.135.306-.315.67-.557 1.094-.728.424-.171.891-.257 1.402-.257.355 0 .688.026.998.076.31.05.606.132.888.246v1.135a3.77 3.77 0 00-.905-.339 4.248 4.248 0 00-.954-.106c-.305 0-.594.058-.865.174-.27.117-.51.284-.714.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.201 1.45.605 1.863.403.412.962.619 1.678.619a3.897 3.897 0 001.818-.438v1.108zm8.538.252h-1.633l-3.2-3.677v3.677h-1.189v-9.659h1.19v5.934l3.082-3.138h1.573l-3.22 3.165 3.397 3.698zm5.066-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 334.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 348.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 342.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1300" fill="#A7333A" d="M113.47 349.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.601-2.713c0 .391-.08.76-.239 1.104-.16.344-.399.644-.718.902-.319.257-.72.46-1.203.608-.483.148-1.046.222-1.688.222a12.014 12.014 0 01-1.75-.123v-1.066a11.124 11.124 0 001.86.157c.446 0 .827-.04 1.144-.12.317-.08.575-.194.776-.345.2-.15.346-.332.437-.546.092-.215.137-.456.137-.725 0-.246-.055-.46-.164-.643a1.365 1.365 0 00-.461-.458 2.281 2.281 0 00-.711-.277 4.16 4.16 0 00-.906-.092h-1.019v-.977h1.033c.269 0 .513-.036.734-.106.222-.07.41-.172.568-.304.157-.133.278-.296.362-.49.084-.193.127-.413.127-.659 0-.478-.146-.827-.438-1.046-.292-.219-.72-.328-1.285-.328-.3 0-.61.03-.93.089-.319.06-.66.148-1.025.267v-1.04c.155-.054.32-.103.496-.147a7.2 7.2 0 011.049-.177c.173-.016.34-.024.499-.024.474 0 .89.051 1.25.154.36.102.662.25.903.44.242.192.424.424.547.698.123.273.185.58.185.923 0 .51-.131.938-.393 1.285-.262.346-.621.622-1.077.827.232.036.461.11.687.219.225.11.428.25.608.42.18.171.326.373.438.605.111.233.167.49.167.773zm8.074-1.75c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]}};" fill="#7E7C7B" d="M31.247 357h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V357zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V357h3.37v12.701zM15.854 374h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V374zM.458 391h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.092.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174H.458v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V391zm7.84 11.3c.233.009.459-.012.678-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm2.14-8.388c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.28-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 37)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-4" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 168)"><path id="Line" d="M.589 2.811h13.948"/><path id="Line-Copy" d="M.589 250.189h13.948"/><path id="Line-Copy-2" d="M15.508 2.811v247.378"/></g><g id="Group-5-Copy-3" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 38 9)"><path id="Line" d="M.589 4.722h13.948"/><path id="Line-Copy" d="M.589 420.278h13.948"/><path id="Line-Copy-2" d="M15.508 4.722v415.556"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 189)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 315)"><path id="Line" d="M.589.844h13.948"/><path id="Line-Copy" d="M.589 75.156h13.948"/><path id="Line-Copy-2" d="M15.508.844v74.312"/></g><path id="Line-Copy-7" fill="#C06334" fill-rule="nonzero" d="M71.5 213.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 213.5z"/><path id="Line-Copy-8" fill="#C06334" fill-rule="nonzero" d="M129.5 323.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 323.5z"/><path id="Line-Copy-9" fill="#C06334" fill-rule="nonzero" d="M129.5 239.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 239.5z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M71.5 110.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 110.5z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="293" height="437" viewBox="0 0 293 437"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="recursive-salaries.svg"><g id="{" fill-rule="nonzero" transform="translate(130.107 17.102)"><path fill="#7E7C7B" d="M5.024 12.701H4.52c-.88 0-1.534-.206-1.962-.618-.429-.413-.643-1.038-.643-1.877V7.854a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096H0v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499C3.635.058 4.04 0 4.52 0h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sales" fill="#181717" d="M20.815 25.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V20.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.157 4.887c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683z"/><path id=":[{" fill="#7E7C7B" d="M56.622 19.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.281-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.281-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm17.274 4.78h-3.37V17h3.37v.957h-2.235v10.787h2.235v.957zm8.101 0h-.506c-.88 0-1.533-.207-1.962-.62-.428-.412-.642-1.037-.642-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.264-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .603-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.246-.427c.046-.173.069-.378.069-.615v-1.62c0-.383.045-.728.136-1.036.092-.307.24-.57.448-.786.208-.216.476-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.058 0-1.586.513-1.586 1.538v1.593c0 1.112-.481 1.738-1.443 1.88.971.095 1.456.72 1.456 1.873v2.338c0 1.043.525 1.565 1.573 1.565h.403v.957z"/><path id="name" fill="#181717" d="M30.775 37.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V37.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V38.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 36.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'John'" fill="#478964" d="M80.5 34.24l-.198 3.205h-1.176l-.191-3.206H80.5zm9.058.724v6.207c0 .383-.057.746-.171 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.35.15-.763.225-1.237.225-.178 0-.359-.012-.544-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.597.367.94.486.345.118.685.178 1.023.178.496 0 .88-.152 1.148-.455.269-.303.403-.74.403-1.31v-5.153h-3.397v-1.053h4.635zm8.77 5.448a4.55 4.55 0 01-.225 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.99-.673 2.98 2.98 0 01-.623-1.1c-.144-.438-.215-.942-.215-1.511 0-.533.075-1.02.225-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.324.23.39.152.72.375.99.669.272.294.48.66.623 1.097.143.438.215.939.215 1.504zm-1.217.055c0-.424-.046-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.356.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.174.212.383.372.63.479.246.107.524.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.356-.81 4.09 4.09 0 00.112-.974zm8.456 3.431h-1.19v-4.381c0-.529-.098-.924-.296-1.186-.199-.262-.482-.394-.852-.394-.159 0-.308.022-.447.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.493.434c-.177.182-.382.41-.615.684v4.552h-1.19V34.24h1.19v2.796l-.04 1.08c.186-.223.37-.411.55-.564.18-.152.36-.277.54-.372.18-.096.363-.164.55-.205.187-.041.38-.062.58-.062.684 0 1.213.209 1.587.626.374.417.56 1.044.56 1.883v4.477zm2.181-6.863h1.06l.047 1.108c.201-.237.395-.435.582-.592.186-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.49.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm11.238-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M123.676 45.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24V45.3z"/><path id="salary" fill="#181717" d="M36.21 59.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369V54.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.407-.83-.407-.16 0-.32.029-.482.086a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.81 6.81 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 53.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="1000" fill="#A7333A" d="M98.075 60.898h-5.612v-1.107h2.297V53.27l-2.14 1.162-.437-1.012 2.843-1.497h1.053v7.868h1.996v1.107zm7.978-4.463c0 .683-.069 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="},{" fill="#7E7C7B" d="M15.853 68h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V68zm7.758 11.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24V79.3zm19.9 1.401h-.506c-.88 0-1.534-.206-1.962-.618-.428-.413-.643-1.038-.643-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.499-.277 2.706 2.706 0 00-.783-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.807-.499.33-.116.734-.174 1.213-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M30.775 88.035h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.386-.896-.386-.146 0-.288.021-.427.064a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.591.684v4.552h-1.19v-6.863zm12.107 6.863l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369V88.35a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.312 3.124V89.97c0-.214-.008-.39-.023-.527a1.2 1.2 0 00-.076-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489c-.107.203-.233.452-.38.748v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.244.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.38-.246.134-.053.283-.079.447-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.11-.237.216-.448.321-.633a2.31 2.31 0 01.338-.465c.121-.125.255-.22.4-.287.146-.066.315-.099.506-.099.862 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M64.32 87.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.558-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="'Alice'" fill="#478964" d="M80.5 85.24l-.198 3.205h-1.176l-.191-3.206H80.5zm10.691 9.658h-1.326l-.622-1.948h-3.719l-.628 1.948H83.63l2.967-8.934h1.668l2.925 8.934zm-2.296-3.035l-1.511-4.778-1.511 4.778h3.022zm5.796-5.646h-2.03v-.978h3.234v8.668h2.043v.991h-5.502v-.99h2.255v-7.691zm7.698 2.803h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256V89.02zm.417-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm10.274 9.489c-.31.118-.628.206-.954.263a5.845 5.845 0 01-1.008.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.879-1.447-.879-2.577 0-.542.085-1.034.253-1.476.169-.442.406-.82.711-1.135.306-.315.67-.557 1.094-.728.424-.171.89-.257 1.401-.257.356 0 .689.026.998.076.31.05.607.132.89.246v1.135a3.77 3.77 0 00-.907-.339 4.248 4.248 0 00-.953-.106c-.306 0-.594.058-.865.174-.271.117-.51.284-.714.503a2.38 2.38 0 00-.486.8c-.118.314-.177.67-.177 1.066 0 .83.201 1.45.604 1.863.404.412.963.619 1.679.619a3.897 3.897 0 001.818-.438v1.108zm8.162-3.542c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm6.679-5.695l-.199 3.206h-1.175l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M131.373 96.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.464 2.238 2.238 0 00-.223-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.149.066.278.168.39.305.112.136.2.307.267.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24V96.3z"/><path id="salary" fill="#181717" d="M36.21 110.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm6.672 1.873l-.028-.922c-.373.369-.753.635-1.138.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .133.021.259.062.38.041.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.768 8.681l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm3.699-3.739h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.521 0l-2.338 6.139a12.037 12.037 0 01-.748 1.613 4.784 4.784 0 01-.844 1.11c-.306.29-.64.503-1.005.64a3.463 3.463 0 01-1.224.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.119.012.244.017.376.017.22 0 .423-.031.612-.095.19-.064.368-.168.537-.311.168-.144.33-.33.485-.557.155-.228.306-.504.451-.828l-2.74-6.863h1.353l1.736 4.54.349 1.066.396-1.094 1.607-4.512h1.305z"/><path id=":" fill="#7E7C7B" d="M79.714 104.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="600" fill="#A7333A" d="M98.26 109.068c0 .41-.074.796-.222 1.156-.149.36-.36.673-.633.94a2.957 2.957 0 01-.988.628 3.45 3.45 0 01-1.281.23c-.492 0-.928-.08-1.306-.236a2.32 2.32 0 01-.95-.725c-.255-.326-.449-.737-.581-1.234-.132-.497-.198-1.084-.198-1.764 0-.455.03-.897.088-1.326.06-.428.158-.831.294-1.21.137-.378.32-.724.547-1.039.228-.314.512-.584.851-.81.34-.225.74-.4 1.2-.526.46-.125.994-.188 1.6-.188h.882v1.039h-.958c-.533 0-.995.064-1.387.191-.392.128-.72.308-.985.54a2.334 2.334 0 00-.608.841c-.141.328-.23.695-.267 1.1l-.027.281c.287-.169.619-.304.995-.407a4.624 4.624 0 011.22-.153c.451 0 .846.066 1.186.198.34.132.622.315.848.55.225.235.395.516.509.844.114.328.17.688.17 1.08zm-1.244.076c0-.274-.033-.521-.1-.742a1.402 1.402 0 00-.31-.56 1.42 1.42 0 00-.537-.356 2.137 2.137 0 00-.776-.127c-.173 0-.349.016-.526.048a3.742 3.742 0 00-.523.134c-.171.056-.334.121-.49.194-.154.073-.295.15-.423.233 0 .588.041 1.079.123 1.473.082.394.204.71.366.947s.361.405.598.506c.237.1.51.15.82.15.26 0 .498-.042.715-.127a1.54 1.54 0 00.56-.372c.157-.164.28-.364.37-.598.088-.235.133-.503.133-.803zm9.037-1.71c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.346-1.05.622-1.453s.621-.716 1.036-.937c.415-.22.898-.331 1.449-.331.479 0 .914.09 1.306.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.068 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.678 23.678 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.231.373.38.516.147.144.316.256.505.335.19.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082l.003.423c.003.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.591.076-.844.226-.253.15-.469.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528z"/><path id="}]," fill="#7E7C7B" d="M15.853 119h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V119zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.222V119h3.37v12.701zm3.986-1.401c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="development" fill="#181717" d="M15.032 159.61c0-.583.08-1.101.24-1.555.159-.453.386-.836.68-1.148.294-.312.647-.55 1.06-.711a3.743 3.743 0 011.377-.243c.218 0 .434.014.646.041.212.027.42.07.625.13v-2.885h1.196v9.66H19.79l-.041-1.3c-.333.484-.693.841-1.08 1.074a2.401 2.401 0 01-1.258.348c-.392 0-.737-.082-1.036-.246a2.104 2.104 0 01-.745-.693 3.33 3.33 0 01-.447-1.077 6 6 0 01-.15-1.395zm1.217-.075c0 .83.122 1.448.366 1.856.244.408.589.612 1.035.612.301 0 .619-.135.954-.403.335-.27.687-.668 1.056-1.197v-3.185a2.892 2.892 0 00-.65-.209 3.55 3.55 0 00-.703-.071c-.647 0-1.152.21-1.515.628-.362.42-.543 1.076-.543 1.97zm12.626-.43c0 .168-.002.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.171a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm2.495-2.899h1.353l1.682 4.54.362 1.107.376-1.135 1.668-4.512h1.306l-2.693 6.863h-1.368l-2.686-6.863zm14.137 3.07c0 .168-.003.31-.007.423a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.195.835.195 1.322zm-1.238-.171a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm5.476-4.717h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.334 5.195a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.031.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.216-.942-.216-1.511 0-.533.076-1.02.226-1.46.15-.44.367-.818.65-1.134a2.913 2.913 0 011.032-.739 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.323.23.39.152.72.375.991.669.272.294.479.66.622 1.097.144.438.216.939.216 1.504zm-1.217.055c0-.424-.047-.794-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.914.208-.252.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .424.047.795.14 1.114.094.32.227.585.4.796.173.212.383.372.63.479.245.107.523.16.833.16.356 0 .66-.069.913-.208.253-.139.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm8.805-.123c0 .61-.086 1.144-.257 1.6-.17.455-.406.833-.707 1.134a2.9 2.9 0 01-1.067.677c-.41.15-.854.225-1.333.225-.218 0-.436-.01-.652-.034a4.952 4.952 0 01-.66-.116v2.871h-1.19v-9.666h1.06l.075 1.149c.342-.47.707-.8 1.094-.988.387-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.545.396.745.694.2.299.351.659.451 1.08.1.422.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.99a2.586 2.586 0 00-.25-.773 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.635-.18c-.15 0-.304.023-.458.07a1.857 1.857 0 00-.482.24 3.59 3.59 0 00-.527.445 7 7 0 00-.591.687v3.329c.219.09.449.163.69.215.242.052.479.079.711.079.643 0 1.146-.218 1.511-.653.365-.435.547-1.09.547-1.966zm7.943 3.5v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.489 30.3 30.3 0 00-.38.748v4.41H71.47v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.134-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.244-.194.379-.246.134-.053.284-.079.448-.079.369 0 .65.12.84.362.192.242.288.616.288 1.122.109-.237.216-.448.32-.633a2.31 2.31 0 01.34-.465c.12-.125.253-.22.4-.287.145-.066.314-.099.505-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.98-.776a2.94 2.94 0 011.32-.287c.479 0 .903.075 1.272.226.369.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.137-2.899h1.06l.048 1.108c.2-.237.394-.435.58-.592.188-.157.371-.283.551-.38a2.06 2.06 0 01.55-.2c.187-.04.38-.059.581-.059.707 0 1.241.209 1.603.626.363.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.489.434s-.373.41-.591.684v4.552h-1.19v-6.863zm13.228 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998H94.8v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.159.365.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025z"/><path id=":{" fill="#7E7C7B" d="M102.806 155.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.677 4.78h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.813-.082.21-.054.374-.138.493-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.09-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.505v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="sites" fill="#181717" d="M36.21 178.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .99-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.069a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683zm4.6-4.005h-2.03v-.985h3.234v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.418-3.863c.132 0 .255.024.369.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm10.472 9.646a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.895.045.329.03.67.08 1.026.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.503.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M72.017 172.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V170h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.429-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 190.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 189.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Peter'" fill="#478964" d="M95.895 187.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.007 3.458c0 .365-.068.726-.205 1.084a2.68 2.68 0 01-.635.96 3.227 3.227 0 01-1.101.687c-.447.176-.98.263-1.6.263h-1.107v3.206h-1.217v-8.934h2.516c.442 0 .866.049 1.271.147.406.098.762.256 1.07.475.308.219.553.501.735.848.182.346.273.767.273 1.264zm-1.264.055c0-.574-.188-1.014-.564-1.32-.376-.305-.901-.457-1.576-.457h-1.244v3.677h1.135c.72 0 1.275-.157 1.664-.471.39-.315.585-.791.585-1.43zm8.907 2.351c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.201 0 .401-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.68.363.934.639.253.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.862 0-1.504-.195-1.928-.584-.424-.39-.636-.988-.636-1.795v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.487c0 .492.13.86.393 1.104.262.243.648.365 1.158.365.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm7.963-3.699c0 .17-.002.31-.006.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.445.61.575 1.005c.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.101-.4.242-.555.425a2.19 2.19 0 00-.375.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.616.69.367.46.537 1.144.51 2.051h-1.203c.013-.601-.074-1.038-.264-1.309-.189-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm10.958-2.796l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id="," fill="#7E7C7B" d="M146.768 198.3c.232.009.458-.012.676-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.032.643-.413.16-.893.24-1.44.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 212.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 206.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="2000" fill="#A7333A" d="M113.545 213.898h-5.886v-1.066l2.31-2.297c.379-.374.687-.697.924-.97.237-.274.421-.523.553-.75a2.28 2.28 0 00.267-.638c.045-.2.068-.417.068-.65 0-.218-.03-.427-.089-.625a1.49 1.49 0 00-.273-.523 1.268 1.268 0 00-.479-.356 1.734 1.734 0 00-.704-.13c-.378 0-.722.085-1.032.253-.31.17-.595.388-.854.657l-.657-.786c.338-.356.726-.64 1.166-.855.44-.214.951-.321 1.535-.321.396 0 .757.06 1.083.178.326.118.607.289.844.512.237.224.42.497.55.82.13.324.195.689.195 1.094 0 .342-.045.659-.136.95a3.646 3.646 0 01-.414.872c-.184.29-.417.59-.697.899-.28.31-.612.65-.995 1.019l-1.62 1.579h4.34v1.134zm7.902-4.463c0 .683-.068 1.306-.205 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.036.937c-.412.221-.896.331-1.452.331-.479 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.642-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453.275-.403.62-.716 1.035-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.643 1.432.15.579.225 1.265.225 2.058zm-1.203.04c0-.154-.006-.308-.017-.46a23.678 23.678 0 00-.038-.449l-3.78 2.81c.069.237.155.456.26.656.105.2.231.373.38.516.147.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.844-.226.253-.15.469-.373.646-.67.178-.296.315-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.081c0 .14.001.282.004.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.591.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.144 1.528zm12.989.04c0 .684-.069 1.307-.206 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.035.936c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.042c0-.155-.005-.31-.017-.462a23.67 23.67 0 00-.037-.448l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="},{" fill="#7E7C7B" d="M31.247 221h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V221zm7.759 11.3c.232.009.458-.012.677-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm19.9 1.401h-.507c-.88 0-1.533-.206-1.962-.618-.428-.413-.642-1.038-.642-1.877v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.332 0 .604-.027.813-.082.21-.054.374-.138.492-.25a.88.88 0 00.247-.427c.045-.173.068-.378.068-.615v-1.62c0-.383.045-.728.137-1.036.09-.307.24-.57.447-.786.208-.216.477-.382.807-.499.33-.116.735-.174 1.213-.174h.506v.957h-.403c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.403v.957z"/><path id="name" fill="#181717" d="M46.17 241.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 240.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Alex'" fill="#478964" d="M95.895 238.24l-.199 3.205h-1.175l-.192-3.206h1.566zm10.69 9.658h-1.325l-.622-1.948h-3.72l-.628 1.948h-1.265l2.967-8.934h1.668l2.926 8.934zm-2.296-3.035l-1.51-4.778-1.511 4.778h3.021zm5.797-5.646h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm11.156 4.887c0 .17-.002.31-.007.424a6.372 6.372 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.933.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.177.84h3.609zm9.324 3.964h-1.579l-1.853-2.618-1.832 2.618h-1.538l2.66-3.445-2.537-3.418h1.525l1.804 2.639 1.771-2.639h1.47l-2.577 3.445 2.686 3.418zm5.052-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 249.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 263.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 257.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1800" fill="#A7333A" d="M113.47 264.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.772-2.242c0 .379-.077.715-.232 1.009a2.13 2.13 0 01-.65.741c-.278.2-.606.353-.984.458a4.594 4.594 0 01-1.23.157c-.488 0-.918-.054-1.29-.164a2.768 2.768 0 01-.932-.458 1.93 1.93 0 01-.759-1.586c0-.546.153-1.018.458-1.415.305-.396.777-.756 1.415-1.08-.583-.296-1.01-.625-1.278-.987a2.038 2.038 0 01-.404-1.248 2.047 2.047 0 01.715-1.552c.24-.21.539-.377.899-.502.36-.126.784-.188 1.271-.188.46 0 .865.049 1.214.147.348.098.641.238.878.42s.415.404.533.663c.119.26.178.55.178.868 0 .52-.145.962-.434 1.327-.29.364-.7.683-1.234.957.264.132.51.278.738.437a3.2 3.2 0 01.595.533c.169.196.3.415.393.657.093.241.14.51.14.806zm-1.415-4.62c0-.406-.142-.708-.427-.907-.285-.198-.685-.297-1.2-.297-.51 0-.908.098-1.193.294-.285.196-.427.49-.427.882a1.175 1.175 0 00.42.916c.144.132.323.262.537.39.214.127.47.257.766.39.515-.242.898-.494 1.148-.756a1.28 1.28 0 00.376-.913zm.123 4.71c0-.17-.027-.334-.082-.493a1.367 1.367 0 00-.294-.472 2.951 2.951 0 00-.567-.465 6.682 6.682 0 00-.903-.478c-.31.15-.57.3-.782.448a2.698 2.698 0 00-.513.45c-.13.153-.222.312-.277.476a1.632 1.632 0 00-.082.52c0 .205.045.386.133.543.09.157.211.29.366.4.155.11.34.191.554.246.214.055.447.082.697.082.242 0 .47-.025.684-.075.214-.05.4-.127.557-.23.157-.102.281-.232.372-.389.092-.157.137-.345.137-.564zm9.195-2.311c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]," fill="#7E7C7B" d="M31.247 272h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V272zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V272h3.37v12.701zm3.986-1.401c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.223-.32.91.91 0 01-.099-.459.915.915 0 01.222-.588.857.857 0 01.288-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.11.136.2.307.266.512.066.205.099.447.099.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.275.27-.62.484-1.032.643-.412.16-.892.24-1.439.24v-.937z"/><path id="internals" fill="#181717" d="M33.113 310.02h-2.03v-.985h3.233v5.872h2.044v.991h-5.503v-.99h2.256v-4.888zm.417-3.863c.132 0 .255.024.37.072a.893.893 0 01.297.202.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.201.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.663.893.893 0 01.297-.202.944.944 0 01.37-.072zm4.943 2.878h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477H42.8v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm13.227 6.768a5.793 5.793 0 01-.834.147 8.52 8.52 0 01-.875.044c-.861 0-1.504-.195-1.928-.584-.423-.39-.635-.988-.635-1.795v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194H51.7v.998h-3.083v3.487c0 .492.131.86.393 1.104.262.243.649.365 1.159.365.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm7.964-3.699c0 .17-.002.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.584.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.226.37.15.681.363.934.639s.444.61.574 1.005c.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.337-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.101-.399.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61zm3.418-2.899h1.087l.034 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051H66.28c.014-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.83-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm7.417 0h1.06l.047 1.108c.2-.237.394-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.55-.2c.188-.04.381-.059.582-.059.706 0 1.24.209 1.603.626.362.417.543 1.044.543 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.2-.258-.499-.387-.895-.387-.146 0-.289.022-.428.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.449-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.355.222.146.055.324.082.534.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm10.794 6.808a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.4.114.8.2 1.196.26.397.059.79.089 1.183.089.57 0 .991-.078 1.264-.233.274-.155.41-.376.41-.663a.812.812 0 00-.064-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.503-.36.868-.49.364-.129.82-.194 1.367-.194.269 0 .567.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.119.747.243 1.025.373.278.13.504.274.677.434.173.16.297.34.372.54.075.2.113.428.113.683z"/><path id=":[{" fill="#7E7C7B" d="M102.806 308.912c.14 0 .275.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.737.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm0 5.01c.14 0 .275.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.737.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085zm17.274 4.78h-3.37V306h3.37v.957h-2.235v10.787h2.235v.957zm8.1 0h-.505c-.88 0-1.534-.207-1.962-.62-.428-.412-.643-1.037-.643-1.876v-2.352a2.18 2.18 0 00-.075-.598.978.978 0 00-.263-.444 1.224 1.224 0 00-.5-.277 2.706 2.706 0 00-.782-.096h-.294v-.95h.294c.333 0 .604-.027.814-.082.21-.054.373-.138.492-.25a.88.88 0 00.246-.427c.046-.173.068-.378.068-.615v-1.62c0-.383.046-.728.137-1.036.091-.307.24-.57.448-.786.207-.216.476-.382.806-.499.33-.116.735-.174 1.214-.174h.506v.957h-.404c-1.057 0-1.586.513-1.586 1.538v1.593c0 1.112-.48 1.738-1.442 1.88.97.095 1.456.72 1.456 1.873v2.338c0 1.043.524 1.565 1.572 1.565h.404v.957z"/><path id="name" fill="#181717" d="M46.17 326.035h1.06l.047 1.108c.2-.237.395-.435.581-.592.187-.157.37-.283.55-.38a2.06 2.06 0 01.551-.2c.187-.04.38-.059.581-.059.706 0 1.24.209 1.603.626.362.417.544 1.044.544 1.883v4.477h-1.19v-4.381c0-.538-.1-.936-.3-1.193-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.435.226c-.15.107-.313.252-.488.434-.176.182-.373.41-.592.684v4.552h-1.19v-6.863zm12.106 6.863l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm8.313 3.124v-4.928c0-.214-.008-.39-.024-.527a1.2 1.2 0 00-.075-.324.348.348 0 00-.13-.168.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.489 30.3 30.3 0 00-.38.748v4.41h-1.087v-4.8c0-.25-.008-.453-.023-.608a1.42 1.42 0 00-.076-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.109.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.246.134-.053.283-.079.447-.079.37 0 .65.12.841.362.192.242.287.616.287 1.122.11-.237.217-.448.322-.633a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.099.506-.099.861 0 1.292.663 1.292 1.99v4.996h-1.094zm8.6-3.794c0 .17-.003.31-.007.424a6.37 6.37 0 01-.02.322h-4.82c0 .701.196 1.24.588 1.616.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.584.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.226.37.15.68.363.934.639.252.276.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.101-.4.242-.554.425a2.19 2.19 0 00-.376.656 3.026 3.026 0 00-.178.84h3.61z"/><path id=":" fill="#7E7C7B" d="M79.714 325.912c.141 0 .276.029.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.403.971.971 0 01-.086.407 1.12 1.12 0 01-.229.331 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.141 0 .276.03.403.086a1.107 1.107 0 01.564.564.978.978 0 01.086.404.971.971 0 01-.086.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.307c-.146 0-.282-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085z"/><path id="'Jack'" fill="#478964" d="M95.895 323.24l-.199 3.205h-1.175l-.192-3.206h1.566zm9.057.724v6.207c0 .383-.057.746-.17 1.09a2.47 2.47 0 01-.52.9 2.49 2.49 0 01-.875.608c-.351.15-.764.225-1.238.225-.177 0-.358-.012-.543-.037a4.8 4.8 0 01-.533-.103 3.619 3.619 0 01-.482-.157 2.214 2.214 0 01-.39-.202v-1.224c.283.206.596.367.94.486.344.118.685.178 1.022.178.497 0 .88-.152 1.149-.455.268-.303.403-.74.403-1.31v-5.153h-3.398v-1.053h4.635zm7.205 8.934l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.257-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.189-.366c-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.223-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.239.24-.305.393-.066.153-.099.325-.099.516 0 .133.02.259.062.38.04.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm8.75 2.872c-.31.118-.627.206-.953.263a5.845 5.845 0 01-1.009.085c-1.084 0-1.92-.294-2.505-.882-.586-.588-.878-1.447-.878-2.577 0-.542.084-1.034.253-1.476.168-.442.405-.82.71-1.135.306-.315.67-.557 1.094-.728.424-.171.891-.257 1.402-.257.355 0 .688.026.998.076.31.05.606.132.888.246v1.135a3.77 3.77 0 00-.905-.339 4.248 4.248 0 00-.954-.106c-.305 0-.594.058-.865.174-.27.117-.51.284-.714.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.201 1.45.605 1.863.403.412.962.619 1.678.619a3.897 3.897 0 001.818-.438v1.108zm8.538.252h-1.633l-3.2-3.677v3.677h-1.189v-9.659h1.19v5.934l3.082-3.138h1.573l-3.22 3.165 3.397 3.698zm5.066-9.659l-.198 3.206h-1.176l-.192-3.206h1.566z"/><path id="," fill="#7E7C7B" d="M139.07 334.3c.233.009.458-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.1-.464 2.238 2.238 0 00-.221-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.111.136.2.307.266.512.066.205.1.447.1.725 0 .378-.07.742-.21 1.09a2.657 2.657 0 01-.621.926c-.276.27-.62.484-1.033.643-.412.16-.892.24-1.439.24v-.937z"/><path id="salary" fill="#181717" d="M51.604 348.025a1.634 1.634 0 01-.458 1.16 2.084 2.084 0 01-.492.378 3.272 3.272 0 01-.598.26c-.212.069-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.198v-1.094c.401.114.8.2 1.197.26.396.059.79.089 1.182.089.57 0 .991-.078 1.265-.233.273-.155.41-.376.41-.663a.812.812 0 00-.065-.331.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.278 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.31 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.76 1.744 1.744 0 01.588-1.264c.214-.196.504-.36.868-.49.365-.129.82-.194 1.367-.194.27 0 .568.015.896.045.328.03.67.08 1.025.153v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.069a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.345c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.119.747.243 1.025.373.278.13.504.274.677.434.173.16.298.34.373.54.075.2.112.428.112.683zm6.672 1.873l-.027-.922c-.374.369-.753.635-1.138.8-.385.163-.79.245-1.214.245-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.397-.62 2.289 2.289 0 01-.126-.772c0-.688.256-1.227.769-1.616.512-.39 1.27-.585 2.273-.585h1.421v-.601c0-.406-.13-.73-.39-.974-.259-.244-.655-.366-1.189-.366-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.642-.1c.224-.024.45-.037.677-.037.415 0 .788.046 1.121.137.333.091.614.23.844.417.23.187.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.059-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .133.02.259.062.38.041.12.107.228.198.321.091.093.21.168.356.222.145.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.117-.789v-1.217zm5.756-5.557h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.99h2.256v-7.691zm9.769 8.681l-.027-.922c-.374.369-.754.635-1.139.8-.385.163-.79.245-1.213.245-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.413 1.603 1.603 0 01-.396-.62 2.289 2.289 0 01-.127-.772c0-.688.256-1.227.77-1.616.512-.39 1.27-.585 2.272-.585h1.422v-.601c0-.406-.13-.73-.39-.974s-.656-.366-1.19-.366c-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.073a7.36 7.36 0 011.087-.301 7.6 7.6 0 01.643-.1c.223-.024.449-.037.677-.037.414 0 .788.046 1.12.137.333.091.615.23.845.417.23.187.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.059-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .133.02.259.061.38.042.12.108.228.199.321.09.093.21.168.355.222.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm3.698-3.739h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.366.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.27-.466-.406-.831-.406-.16 0-.32.028-.482.085a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H77.24v-6.864zm13.522 0l-2.338 6.139a12.037 12.037 0 01-.749 1.613 4.784 4.784 0 01-.844 1.11c-.305.29-.64.503-1.005.64a3.463 3.463 0 01-1.223.205 6.811 6.811 0 01-.636-.027v-1.08c.1.013.21.026.328.037.118.012.244.017.376.017.219 0 .423-.031.612-.095s.368-.168.536-.311c.169-.144.33-.33.486-.557.155-.228.305-.504.45-.828l-2.74-6.863h1.353l1.737 4.54.348 1.066.397-1.094 1.606-4.512h1.306z"/><path id=":" fill="#7E7C7B" d="M95.108 342.912c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.145 0-.281-.027-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/><path id="1300" fill="#A7333A" d="M113.47 349.898h-5.613v-1.107h2.297v-6.521l-2.14 1.162-.437-1.012 2.844-1.497h1.053v7.868h1.996v1.107zm7.601-2.713c0 .391-.08.76-.239 1.104-.16.344-.399.644-.718.902-.319.257-.72.46-1.203.608-.483.148-1.046.222-1.688.222a12.014 12.014 0 01-1.75-.123v-1.066a11.124 11.124 0 001.86.157c.446 0 .827-.04 1.144-.12.317-.08.575-.194.776-.345.2-.15.346-.332.437-.546.092-.215.137-.456.137-.725 0-.246-.055-.46-.164-.643a1.365 1.365 0 00-.461-.458 2.281 2.281 0 00-.711-.277 4.16 4.16 0 00-.906-.092h-1.019v-.977h1.033c.269 0 .513-.036.734-.106.222-.07.41-.172.568-.304.157-.133.278-.296.362-.49.084-.193.127-.413.127-.659 0-.478-.146-.827-.438-1.046-.292-.219-.72-.328-1.285-.328-.3 0-.61.03-.93.089-.319.06-.66.148-1.025.267v-1.04c.155-.054.32-.103.496-.147a7.2 7.2 0 011.049-.177c.173-.016.34-.024.499-.024.474 0 .89.051 1.25.154.36.102.662.25.903.44.242.192.424.424.547.698.123.273.185.58.185.923 0 .51-.131.938-.393 1.285-.262.346-.621.622-1.077.827.232.036.461.11.687.219.225.11.428.25.608.42.18.171.326.373.438.605.111.233.167.49.167.773zm8.074-1.75c0 .683-.069 1.306-.206 1.87a4.33 4.33 0 01-.622 1.448 2.925 2.925 0 01-1.035.937c-.413.221-.897.331-1.453.331-.478 0-.914-.09-1.306-.27a2.594 2.594 0 01-1.004-.83c-.278-.374-.493-.85-.643-1.429-.15-.579-.226-1.264-.226-2.057 0-.684.07-1.308.209-1.873.139-.566.346-1.05.622-1.453s.62-.716 1.036-.937c.414-.22.897-.331 1.449-.331.478 0 .914.09 1.305.27.392.18.727.458 1.005.834.278.376.493.853.643 1.432.15.579.226 1.265.226 2.058zm-1.204.04c0-.154-.005-.308-.017-.46a23.67 23.67 0 00-.037-.449l-3.78 2.81c.068.237.154.456.26.656.104.2.23.373.379.516.148.144.316.256.505.335.19.08.405.12.646.12.31 0 .592-.075.845-.226.253-.15.468-.373.646-.67.177-.296.314-.664.41-1.104.096-.44.143-.949.143-1.527zm-4.087-.081c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.376-.496 1.612 1.612 0 00-.499-.324 1.628 1.628 0 00-.629-.117c-.31 0-.59.076-.844.226-.253.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528zm12.988.04c0 .684-.069 1.307-.205 1.87a4.33 4.33 0 01-.622 1.45 2.925 2.925 0 01-1.036.936c-.412.221-.897.331-1.453.331-.478 0-.913-.09-1.305-.27a2.594 2.594 0 01-1.005-.83c-.278-.374-.492-.85-.643-1.429-.15-.579-.225-1.264-.225-2.057 0-.684.07-1.308.208-1.873.14-.566.347-1.05.622-1.453.276-.403.621-.716 1.036-.937.415-.22.898-.331 1.45-.331.478 0 .913.09 1.305.27.392.18.727.458 1.005.834.278.376.492.853.642 1.432.15.579.226 1.265.226 2.058zm-1.203.042c0-.155-.006-.31-.017-.462a23.67 23.67 0 00-.038-.448l-3.78 2.81c.068.237.155.456.26.656.104.2.23.373.379.516.148.144.317.256.506.335.189.08.404.12.646.12.31 0 .591-.075.844-.226.253-.15.468-.373.646-.67.178-.296.314-.664.41-1.104.096-.44.144-.949.144-1.527zm-4.088-.082c0 .14 0 .282.003.423.002.142.01.278.024.41l3.78-2.795a3.198 3.198 0 00-.26-.636 2.016 2.016 0 00-.375-.496 1.612 1.612 0 00-.5-.324 1.628 1.628 0 00-.628-.117c-.31 0-.592.076-.845.226-.252.15-.468.375-.646.673a3.697 3.697 0 00-.41 1.108 7.24 7.24 0 00-.143 1.528z"/><path id="}]}};" fill="#7E7C7B" d="M31.247 357h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.091.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174h-.506v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V357zm11.47 12.701h-3.37v-.957h2.222v-10.787h-2.221V357h3.37v12.701zM15.854 374h.505c.88 0 1.534.206 1.962.619.429.412.643 1.038.643 1.876v1.58c0 .223.025.422.075.597.05.176.138.324.263.445.126.12.293.213.503.277.21.063.471.095.786.095h.294v.95h-.294c-.333 0-.605.028-.817.082a1.116 1.116 0 00-.496.25.88.88 0 00-.246.427 2.438 2.438 0 00-.068.615v2.393c0 .383-.046.728-.137 1.036-.091.307-.24.57-.444.786a1.993 1.993 0 01-.807.499c-.332.116-.738.174-1.217.174h-.505v-.957h.41c1.052 0 1.579-.513 1.579-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.97-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.565-1.565h-.41V374zM.458 391h.506c.88 0 1.533.206 1.962.619.428.412.642 1.038.642 1.876v1.58c0 .223.025.422.076.597.05.176.137.324.263.445.125.12.293.213.502.277.21.063.472.095.786.095h.294v.95h-.294c-.332 0-.605.028-.817.082a1.116 1.116 0 00-.495.25.88.88 0 00-.246.427 2.438 2.438 0 00-.069.615v2.393c0 .383-.045.728-.136 1.036-.092.307-.24.57-.445.786a1.993 1.993 0 01-.806.499c-.333.116-.739.174-1.217.174H.458v-.957h.41c1.053 0 1.58-.513 1.58-1.538v-2.365c0-1.112.48-1.737 1.442-1.873-.971-.105-1.456-.732-1.456-1.88v-1.566c0-1.043-.522-1.565-1.566-1.565h-.41V391zm7.84 11.3c.233.009.459-.012.678-.062a2.02 2.02 0 00.577-.222c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.464 2.238 2.238 0 00-.222-.329 2.438 2.438 0 01-.222-.32.91.91 0 01-.1-.459.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.305.112.136.2.307.266.512.067.205.1.447.1.725 0 .378-.07.742-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.484-1.032.643-.413.16-.892.24-1.44.24v-.937zm2.14-8.388c.142 0 .276.029.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.331 1.058 1.058 0 01-.738.308c-.145 0-.28-.028-.406-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.407-.085z"/></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 37)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-4" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 105 168)"><path id="Line" d="M.589 2.811h13.948"/><path id="Line-Copy" d="M.589 250.189h13.948"/><path id="Line-Copy-2" d="M15.508 2.811v247.378"/></g><g id="Group-5-Copy-3" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 38 9)"><path id="Line" d="M.589 4.722h13.948"/><path id="Line-Copy" d="M.589 420.278h13.948"/><path id="Line-Copy-2" d="M15.508 4.722v415.556"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 189)"><path id="Line" d="M.589 1.244h13.948"/><path id="Line-Copy" d="M.589 110.756h13.948"/><path id="Line-Copy-2" d="M15.508 1.244v109.512"/></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 153 315)"><path id="Line" d="M.589.844h13.948"/><path id="Line-Copy" d="M.589 75.156h13.948"/><path id="Line-Copy-2" d="M15.508.844v74.312"/></g><path id="Line-Copy-7" fill="#C06334" fill-rule="nonzero" d="M71.5 213.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 213.5z" transform="matrix(1 0 0 -1 0 499)"/><path id="Line-Copy-8" fill="#C06334" fill-rule="nonzero" d="M129.5 323.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 323.5z" transform="matrix(1 0 0 -1 0 670)"/><path id="Line-Copy-9" fill="#C06334" fill-rule="nonzero" d="M129.5 239.5l-5.848 14.519-3.976-4.494-15.513 13.724-.75.663-1.325-1.498.75-.663 15.513-13.724-3.975-4.494L129.5 239.5z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M71.5 110.5l-.344 15.649-5.307-2.801-31.465 59.619-.466.884-1.77-.933.468-.885 31.464-59.619-5.305-2.8L71.5 110.5z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/06-advanced-functions/02-rest-parameters-spread-operator/article.md b/1-js/06-advanced-functions/02-rest-parameters-spread/article.md similarity index 68% rename from 1-js/06-advanced-functions/02-rest-parameters-spread-operator/article.md rename to 1-js/06-advanced-functions/02-rest-parameters-spread/article.md index a98d8eddd..dbdfbd6c0 100644 --- a/1-js/06-advanced-functions/02-rest-parameters-spread-operator/article.md +++ b/1-js/06-advanced-functions/02-rest-parameters-spread/article.md @@ -1,4 +1,4 @@ -# Rest parameters and spread operator +# Rest parameters and spread syntax Many JavaScript built-in functions support an arbitrary number of arguments. @@ -8,7 +8,7 @@ For instance: - `Object.assign(dest, src1, ..., srcN)` -- copies properties from `src1..N` into `dest`. - ...and so on. -In this chapter we'll learn how to do the same. And, more importantly, how to feel comfortable working with such functions and arrays. +In this chapter we'll learn how to do the same. And also, how to pass arrays to such functions as parameters. ## Rest parameters `...` @@ -23,9 +23,9 @@ function sum(a, b) { alert( sum(1, 2, 3, 4, 5) ); ``` -There will be no error because of "excessive" arguments. But of course in the result only the first two will be counted. +There will be no error because of "excessive" arguments. But of course in the result only the first two will be counted, so the result in the code above is `3`. -The rest parameters can be mentioned in a function definition with three dots `...`. They literally mean "gather the remaining parameters into an array". +The rest of the parameters can be included in the function definition by using three dots `...` followed by the name of the array that will contain them. The dots literally mean "gather the remaining parameters into an array". For instance, to gather all arguments into array `args`: @@ -96,9 +96,7 @@ showName("Julius", "Caesar"); showName("Ilya"); ``` -In old times, rest parameters did not exist in the language, and using `arguments` was the only way to get all arguments of the function, no matter their total number. - -And it still works, we can use it today. +In old times, rest parameters did not exist in the language, and using `arguments` was the only way to get all arguments of the function. And it still works, we can find it in the old code. But the downside is that although `arguments` is both array-like and iterable, it's not an array. It does not support array methods, so we can't call `arguments.map(...)` for example. @@ -119,11 +117,12 @@ function f() { f(1); // 1 ``` -```` As we remember, arrow functions don't have their own `this`. Now we know they don't have the special `arguments` object either. +```` + -## Spread operator [#spread-operator] +## Spread syntax [#spread-syntax] We've just seen how to get an array from the list of parameters. @@ -149,7 +148,7 @@ alert( Math.max(arr) ); // NaN And surely we can't manually list items in the code `Math.max(arr[0], arr[1], arr[2])`, because we may be unsure how many there are. As our script executes, there could be a lot, or there could be none. And that would get ugly. -*Spread operator* to the rescue! It looks similar to rest parameters, also using `...`, but does quite the opposite. +*Spread syntax* to the rescue! It looks similar to rest parameters, also using `...`, but does quite the opposite. When `...arr` is used in the function call, it "expands" an iterable object `arr` into the list of arguments. @@ -170,7 +169,7 @@ let arr2 = [8, 3, -8, 1]; alert( Math.max(...arr1, ...arr2) ); // 8 ``` -We can even combine the spread operator with normal values: +We can even combine the spread syntax with normal values: ```js run @@ -180,7 +179,7 @@ let arr2 = [8, 3, -8, 1]; alert( Math.max(1, ...arr1, 2, ...arr2, 25) ); // 25 ``` -Also, the spread operator can be used to merge arrays: +Also, the spread syntax can be used to merge arrays: ```js run let arr = [3, 5, 1]; @@ -193,9 +192,9 @@ let merged = [0, ...arr, 2, ...arr2]; alert(merged); // 0,3,5,1,2,8,9,15 (0, then arr, then 2, then arr2) ``` -In the examples above we used an array to demonstrate the spread operator, but any iterable will do. +In the examples above we used an array to demonstrate the spread syntax, but any iterable will do. -For instance, here we use the spread operator to turn the string into array of characters: +For instance, here we use the spread syntax to turn the string into array of characters: ```js run let str = "Hello"; @@ -203,7 +202,7 @@ let str = "Hello"; alert( [...str] ); // H,e,l,l,o ``` -The spread operator internally uses iterators to gather elements, the same way as `for..of` does. +The spread syntax internally uses iterators to gather elements, the same way as `for..of` does. So, for a string, `for..of` returns characters and `...str` becomes `"H","e","l","l","o"`. The list of characters is passed to array initializer `[...str]`. @@ -221,24 +220,75 @@ The result is the same as `[...str]`. But there's a subtle difference between `Array.from(obj)` and `[...obj]`: - `Array.from` operates on both array-likes and iterables. -- The spread operator operates only on iterables. +- The spread syntax works only with iterables. So, for the task of turning something into an array, `Array.from` tends to be more universal. +## Copy an array/object + +Remember when we talked about `Object.assign()` [in the past](info:object-copy#cloning-and-merging-object-assign)? + +It is possible to do the same thing with the spread syntax. + +```js run +let arr = [1, 2, 3]; + +*!* +let arrCopy = [...arr]; // spread the array into a list of parameters + // then put the result into a new array +*/!* + +// do the arrays have the same contents? +alert(JSON.stringify(arr) === JSON.stringify(arrCopy)); // true + +// are the arrays equal? +alert(arr === arrCopy); // false (not same reference) + +// modifying our initial array does not modify the copy: +arr.push(4); +alert(arr); // 1, 2, 3, 4 +alert(arrCopy); // 1, 2, 3 +``` + +Note that it is possible to do the same thing to make a copy of an object: + +```js run +let obj = { a: 1, b: 2, c: 3 }; + +*!* +let objCopy = { ...obj }; // spread the object into a list of parameters + // then return the result in a new object +*/!* + +// do the objects have the same contents? +alert(JSON.stringify(obj) === JSON.stringify(objCopy)); // true + +// are the objects equal? +alert(obj === objCopy); // false (not same reference) + +// modifying our initial object does not modify the copy: +obj.d = 4; +alert(JSON.stringify(obj)); // {"a":1,"b":2,"c":3,"d":4} +alert(JSON.stringify(objCopy)); // {"a":1,"b":2,"c":3} +``` + +This way of copying an object is much shorter than `let objCopy = Object.assign({}, obj)` or for an array `let arrCopy = Object.assign([], arr)` so we prefer to use it whenever we can. + + ## Summary -When we see `"..."` in the code, it is either rest parameters or the spread operator. +When we see `"..."` in the code, it is either rest parameters or the spread syntax. There's an easy way to distinguish between them: - When `...` is at the end of function parameters, it's "rest parameters" and gathers the rest of the list of arguments into an array. -- When `...` occurs in a function call or alike, it's called a "spread operator" and expands an array into a list. +- When `...` occurs in a function call or alike, it's called a "spread syntax" and expands an array into a list. Use patterns: - Rest parameters are used to create functions that accept any number of arguments. -- The spread operator is used to pass an array to functions that normally require a list of many arguments. +- The spread syntax is used to pass an array to functions that normally require a list of many arguments. Together they help to travel between a list and an array of parameters with ease. diff --git a/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/solution.md b/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/solution.md new file mode 100644 index 000000000..7cbd85ab7 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/solution.md @@ -0,0 +1,5 @@ +The answer is: **Pete**. + +A function gets outer variables as they are now, it uses the most recent values. + +Old variable values are not saved anywhere. When a function wants a variable, it takes the current value from its own Lexical Environment or the outer one. diff --git a/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/task.md b/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/task.md new file mode 100644 index 000000000..819189773 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/1-closure-latest-changes/task.md @@ -0,0 +1,23 @@ +importance: 5 + +--- + +# Does a function pickup latest changes? + +The function sayHi uses an external variable name. When the function runs, which value is it going to use? + +```js +let name = "John"; + +function sayHi() { + alert("Hi, " + name); +} + +name = "Pete"; + +sayHi(); // what will it show: "John" or "Pete"? +``` + +Such situations are common both in browser and server-side development. A function may be scheduled to execute later than it is created, for instance after a user action or a network request. + +So, the question is: does it pick up the latest changes? diff --git a/1-js/06-advanced-functions/03-closure/8-make-army/_js.view/solution.js b/1-js/06-advanced-functions/03-closure/10-make-army/_js.view/solution.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/8-make-army/_js.view/solution.js rename to 1-js/06-advanced-functions/03-closure/10-make-army/_js.view/solution.js diff --git a/1-js/06-advanced-functions/03-closure/8-make-army/_js.view/source.js b/1-js/06-advanced-functions/03-closure/10-make-army/_js.view/source.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/8-make-army/_js.view/source.js rename to 1-js/06-advanced-functions/03-closure/10-make-army/_js.view/source.js diff --git a/1-js/06-advanced-functions/03-closure/8-make-army/_js.view/test.js b/1-js/06-advanced-functions/03-closure/10-make-army/_js.view/test.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/8-make-army/_js.view/test.js rename to 1-js/06-advanced-functions/03-closure/10-make-army/_js.view/test.js diff --git a/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-empty.svg b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-empty.svg new file mode 100644 index 000000000..f8c7bd6ac --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-empty.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="566" height="183" viewBox="0 0 566 183"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="lexenv-makearmy-empty.svg"><g id="shooters-=-[" fill-rule="nonzero" transform="translate(6.176 15.2)"><path id="shooters" fill="#181717" d="M4.956 7.91a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315A3.002 3.002 0 010 8.988l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zM7.28 0h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V9.8h-1.092V5.978c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V9.8H8.386V.938H7.28V0zm9.142 6.3c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.238 0c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm6.874-3.5h1.666V1.414l1.092-.308V2.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.738h-1.666V2.8zM48.23 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM50.148 9.8v-.938h1.89V3.738h-1.89V2.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688V9.8h-5.656zm13.608-1.89a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525z"/><path id="=" fill="#DBAF88" d="M75.376 3.402h6.496V4.41h-6.496V3.402zm0 2.296h6.496v1.008h-6.496V5.698z"/><path id="[" fill="#7E7C7B" d="M93.31 0h4.774v.98h-3.682v11.06h3.682v.98H93.31z"/><path id="function" fill="#1C85B5" d="M16.856 33.8v-.938h1.666v-5.124h-1.666V26.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V26.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V26.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V33.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V33.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V26.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V26.8zm9.436 7v-.938h2.436v-5.124h-2.436V26.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V33.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V33.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 36.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 24l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 27.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 24h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V24zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V26.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V26.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V26.8z"/><path id="(" fill="#7E7C7B" d="M181.272 36.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 33.8v-.938h2.436v-5.124h-2.436V26.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 35.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V33.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V31c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="function" fill="#1C85B5" d="M16.856 57.8v-.938h1.666v-5.124h-1.666V50.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V50.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V50.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V57.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V57.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V50.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V50.8zm9.436 7v-.938h2.436v-5.124h-2.436V50.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V57.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V57.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 60.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 48l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 51.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 48h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V48zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V50.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V50.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V50.8z"/><path id="(" fill="#7E7C7B" d="M181.272 60.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 57.8v-.938h2.436v-5.124h-2.436V50.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 59.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V57.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V55c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="function" fill="#1C85B5" d="M16.856 81.8v-.938h1.666v-5.124h-1.666V74.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V74.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V74.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V81.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V81.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V74.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V74.8zm9.436 7v-.938h2.436v-5.124h-2.436V74.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V81.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V81.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 84.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 72l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 75.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 72h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V72zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V74.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V74.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V74.8z"/><path id="(" fill="#7E7C7B" d="M181.272 84.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 81.8v-.938h2.436v-5.124h-2.436V74.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 83.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V81.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V79c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="..." fill="#181717" d="M18.886 105.03c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm8.4 0c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm8.4 0c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="function" fill="#1C85B5" d="M16.856 129.8v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 132.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 123.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 120h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V120zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="(" fill="#7E7C7B" d="M181.272 132.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 129.8v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}];" fill="#7E7C7B" d="M194.026 131.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zM5.138 157.02H.35v-.98h3.696v-11.06H.35V144h4.788v13.02zm5.292-3.99c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><g id="Group-Copy" transform="translate(366 66)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-Copy-2" fill="#C06334" fill-rule="nonzero" transform="translate(242 44)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-3" transform="translate(279 36)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v18H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14"><empty></tspan></text></g><g id="Group-Copy-3" fill="#C06334" fill-rule="nonzero" transform="translate(242 67)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-Copy-4" fill="#C06334" fill-rule="nonzero" transform="translate(242 90)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-Copy-5" fill="#C06334" fill-rule="nonzero" transform="translate(242 139)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><text id="makeArmy()-LexicalEn" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="415" y="59">makeArmy()</tspan> <tspan x="415" y="74" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text><text id="while-iteration-Lexi" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="278" y="14">while iteration</tspan> <tspan x="278" y="29" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text><g id="Group-3" transform="translate(279 59)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v18H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14"><empty></tspan></text></g><g id="Group-3" transform="translate(279 82)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v18H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14"><empty></tspan></text></g><g id="Group-3" transform="translate(279 131)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v18H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14"><empty></tspan></text></g><g id="Group-3" transform="translate(415 78)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h53v18H1z"/><text id="i:-10" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="6" y="14">i: 10</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-for-fixed.svg b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-for-fixed.svg new file mode 100644 index 000000000..7611d0ef8 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-for-fixed.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="566" height="183" viewBox="0 0 566 183"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="lexenv-makearmy-for-fixed.svg"><g id="shooters-=-[" fill-rule="nonzero" transform="translate(6.176 15.2)"><path id="shooters" fill="#181717" d="M4.956 7.91a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315A3.002 3.002 0 010 8.988l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zM7.28 0h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V9.8h-1.092V5.978c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V9.8H8.386V.938H7.28V0zm9.142 6.3c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.238 0c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm6.874-3.5h1.666V1.414l1.092-.308V2.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.738h-1.666V2.8zM48.23 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM50.148 9.8v-.938h1.89V3.738h-1.89V2.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688V9.8h-5.656zm13.608-1.89a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525z"/><path id="=" fill="#DBAF88" d="M75.376 3.402h6.496V4.41h-6.496V3.402zm0 2.296h6.496v1.008h-6.496V5.698z"/><path id="[" fill="#7E7C7B" d="M93.31 0h4.774v.98h-3.682v11.06h3.682v.98H93.31z"/><path id="function" fill="#1C85B5" d="M16.856 33.8v-.938h1.666v-5.124h-1.666V26.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V26.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V26.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V33.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V33.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V26.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V26.8zm9.436 7v-.938h2.436v-5.124h-2.436V26.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V33.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V33.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 36.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 24l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 27.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 24h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V24zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V26.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V26.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V26.8z"/><path id="(" fill="#7E7C7B" d="M181.272 36.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 33.8v-.938h2.436v-5.124h-2.436V26.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 35.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V33.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V31c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="function" fill="#1C85B5" d="M16.856 57.8v-.938h1.666v-5.124h-1.666V50.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V50.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V50.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V57.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V57.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V50.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V50.8zm9.436 7v-.938h2.436v-5.124h-2.436V50.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V57.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V57.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 60.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 48l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 51.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 48h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V48zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V50.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V50.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V50.8z"/><path id="(" fill="#7E7C7B" d="M181.272 60.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 57.8v-.938h2.436v-5.124h-2.436V50.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 59.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V57.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V55c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="function" fill="#1C85B5" d="M16.856 81.8v-.938h1.666v-5.124h-1.666V74.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V74.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V74.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V81.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V81.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V74.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V74.8zm9.436 7v-.938h2.436v-5.124h-2.436V74.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V81.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V81.8h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 84.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0197.272 72l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 75.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 72h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V72zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V74.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V74.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V74.8z"/><path id="(" fill="#7E7C7B" d="M181.272 84.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 81.8v-.938h2.436v-5.124h-2.436V74.8h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}," fill="#7E7C7B" d="M194.026 83.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V81.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V79c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="..." fill="#181717" d="M18.886 105.03c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm8.4 0c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm8.4 0c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="function" fill="#1C85B5" d="M16.856 129.8v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938h-2.898v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946H46.97v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M97.272 132.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M134.736 123.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM143.29 120h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V120zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="(" fill="#7E7C7B" d="M181.272 132.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="i" fill="#181717" d="M185.094 129.8v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616z"/><path id=");}];" fill="#7E7C7B" d="M194.026 131.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zm18.788 7.196c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zM5.138 157.02H.35v-.98h3.696v-11.06H.35V144h4.788v13.02zm5.292-3.99c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><g id="Group-Copy" transform="translate(350 66)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-Copy-2" fill="#C06334" fill-rule="nonzero" transform="translate(242 44)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-Copy-3" fill="#C06334" fill-rule="nonzero" transform="translate(242 67)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-Copy-4" fill="#C06334" fill-rule="nonzero" transform="translate(242 90)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-Copy-5" fill="#C06334" fill-rule="nonzero" transform="translate(242 139)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><text id="makeArmy()-LexicalEn" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="399" y="59">makeArmy()</tspan> <tspan x="399" y="74" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text><text id="for-iteration-Lexica" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="278" y="14">for iteration</tspan> <tspan x="278" y="29" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text><g id="Group-3" transform="translate(279 36)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="i:-0" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">i: 0</tspan></text></g><g id="Group-3-Copy" transform="translate(279 59)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="i:-1" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">i: 1</tspan></text></g><g id="Group-3-Copy-2" transform="translate(279 82)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="i:-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">i: 2</tspan></text></g><g id="Group-3-Copy-3" transform="translate(279 131)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="i:-10" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7.447" y="13.51">i: 10</tspan></text></g><text id="..." fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="399" y="94">...</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-while-fixed.svg b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-while-fixed.svg new file mode 100644 index 000000000..d83ecbe76 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy-while-fixed.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="566" height="183" viewBox="0 0 566 183"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="lexenv-makearmy-while-fixed.svg"><g id="shooters-=-[-functio" fill-rule="nonzero" transform="translate(7.176 15.2)"><path id="shooters" fill="#181717" d="M2.996 9.968c1.017 0 1.78-.21 2.289-.63.509-.42.763-.952.763-1.596 0-.43-.117-.77-.35-1.022a2.536 2.536 0 00-.868-.602 5.479 5.479 0 00-1.127-.343 19.87 19.87 0 01-1.127-.252 3.633 3.633 0 01-.868-.329c-.233-.13-.35-.317-.35-.56 0-.196.049-.362.147-.497.098-.135.229-.245.392-.329.163-.084.345-.147.546-.189.2-.042.404-.063.609-.063.541 0 .992.077 1.351.231.36.154.693.315 1.001.483l.448-.882c-.28-.177-.656-.348-1.127-.511-.471-.163-1.034-.245-1.687-.245-.364 0-.714.04-1.05.119-.336.08-.63.2-.882.364a1.94 1.94 0 00-.609.623c-.154.252-.231.555-.231.91 0 .43.117.763.35 1.001.233.238.523.427.868.567.345.14.721.252 1.127.336.406.084.782.175 1.127.273.345.098.635.229.868.392.233.163.35.394.35.693 0 .205-.049.38-.147.525a1.08 1.08 0 01-.399.35 2.075 2.075 0 01-.581.196 3.76 3.76 0 01-1.484-.021 4.197 4.197 0 01-1.337-.539 2.536 2.536 0 01-.448-.343l-.56.91c.14.121.317.24.532.357.215.117.453.222.714.315.261.093.541.168.84.224.299.056.602.084.91.084zM9.464 9.8V5.348c.056-.261.156-.502.301-.721.145-.22.317-.406.518-.56.2-.154.422-.275.665-.364.243-.089.49-.133.742-.133.364 0 .663.051.896.154.233.103.418.254.553.455.135.2.226.45.273.749.047.299.07.649.07 1.05V9.8h1.092V5.768c0-.625-.051-1.141-.154-1.547-.103-.406-.259-.726-.469-.959a1.676 1.676 0 00-.812-.49 4.338 4.338 0 00-1.169-.14c-.57 0-1.045.091-1.428.273a3.135 3.135 0 00-1.008.763h-.07V0H7.28v.938h1.106V9.8h1.078zm10.36.168a3.82 3.82 0 001.449-.259 2.907 2.907 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm8.4.938a3.82 3.82 0 001.449-.259 2.907 2.907 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm9.058.952c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512V3.738h3.738V2.8h-3.738V1.106l-1.092.308V2.8h-1.666v.938h1.666V7.21c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161zm7.868-.014c.299 0 .595-.026.889-.077.294-.051.576-.124.847-.217.27-.093.52-.205.749-.336.229-.13.427-.27.595-.42l-.434-.798a2.585 2.585 0 01-.399.28 3.72 3.72 0 01-.595.28c-.224.084-.46.156-.707.217-.247.06-.497.091-.749.091-.756 0-1.36-.208-1.813-.623-.453-.415-.679-1.043-.679-1.883h5.418c.047-.793-.01-1.44-.168-1.939-.159-.5-.385-.891-.679-1.176a2.295 2.295 0 00-1.015-.581 4.466 4.466 0 00-1.162-.154c-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .56.075 1.064.224 1.512.15.448.369.833.658 1.155.29.322.649.57 1.078.742.43.173.924.259 1.484.259zm2.1-4.466h-4.354c.037-.364.124-.67.259-.917s.308-.448.518-.602c.21-.154.453-.264.728-.329a3.78 3.78 0 01.875-.098c.57 0 1.029.17 1.379.511.35.34.548.819.595 1.435zM55.804 9.8v-.938h-2.688v-4.13c.056-.084.145-.187.266-.308.121-.121.268-.243.441-.364s.376-.224.609-.308c.233-.084.495-.126.784-.126.308 0 .52.14.637.42.117.28.175.719.175 1.316l.98-.014c0-.42-.019-.796-.056-1.127a2.771 2.771 0 00-.217-.847 1.206 1.206 0 00-.448-.532c-.191-.121-.446-.182-.763-.182-.541 0-1.008.103-1.4.308a5.046 5.046 0 00-1.064.742h-.07l-.224-.91h-2.618v.938h1.89v5.124h-1.89V9.8h5.656zm5.992.168c1.017 0 1.78-.21 2.289-.63.509-.42.763-.952.763-1.596 0-.43-.117-.77-.35-1.022a2.536 2.536 0 00-.868-.602 5.479 5.479 0 00-1.127-.343 19.87 19.87 0 01-1.127-.252 3.633 3.633 0 01-.868-.329c-.233-.13-.35-.317-.35-.56 0-.196.049-.362.147-.497.098-.135.229-.245.392-.329.163-.084.345-.147.546-.189.2-.042.404-.063.609-.063.541 0 .992.077 1.351.231.36.154.693.315 1.001.483l.448-.882c-.28-.177-.656-.348-1.127-.511-.471-.163-1.034-.245-1.687-.245-.364 0-.714.04-1.05.119-.336.08-.63.2-.882.364a1.94 1.94 0 00-.609.623c-.154.252-.231.555-.231.91 0 .43.117.763.35 1.001.233.238.523.427.868.567.345.14.721.252 1.127.336.406.084.782.175 1.127.273.345.098.635.229.868.392.233.163.35.394.35.693 0 .205-.049.38-.147.525a1.08 1.08 0 01-.399.35 2.075 2.075 0 01-.581.196 3.76 3.76 0 01-1.484-.021 4.197 4.197 0 01-1.337-.539 2.536 2.536 0 01-.448-.343l-.56.91c.14.121.317.24.532.357.215.117.453.222.714.315.261.093.541.168.84.224.299.056.602.084.91.084z"/><path id="=" fill="#DBAF88" d="M81.872 4.41V3.402h-6.496V4.41h6.496zm0 2.296V5.698h-6.496v1.008h6.496z"/><path id="[" fill="#7E7C7B" d="M98.084 13.02v-.98h-3.682V.98h3.682V0H93.31v13.02z"/><path id="function" fill="#1C85B5" d="M22.512 33.8v-.938h-2.898v-5.124h2.898V26.8h-2.898c0-.69.112-1.188.336-1.491.224-.303.644-.455 1.26-.455.187 0 .392.016.616.049.224.033.457.1.7.203l.252-.91a3.603 3.603 0 00-.784-.224 5.473 5.473 0 00-.826-.056c-.85 0-1.503.196-1.96.588-.457.392-.686 1.027-.686 1.904v.392h-1.666v.938h1.666v5.124h-1.666v.938h5.656zm5.222.168c.57 0 1.055-.126 1.456-.378s.723-.579.966-.98h.056l.042 1.19h1.862v-.91h-.938a9.956 9.956 0 01-.084-1.316V26.8h-1.988v.938h.896v3.626c-.187.476-.46.873-.819 1.19-.36.317-.782.476-1.267.476-.336 0-.604-.054-.805-.161a1.046 1.046 0 01-.455-.476 2.317 2.317 0 01-.203-.77 9.693 9.693 0 01-.049-1.029V26.8h-2.03v.938h.938v3.08c0 .607.047 1.113.14 1.519.093.406.238.73.434.973s.446.413.749.511c.303.098.67.147 1.099.147zm7.07-.168v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V33.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078zm10.556.168c.364 0 .702-.033 1.015-.098a4.708 4.708 0 002.023-.952l-.49-.812c-.28.243-.628.439-1.043.588-.415.15-.842.224-1.281.224-.41 0-.786-.06-1.127-.182a2.517 2.517 0 01-.875-.518 2.248 2.248 0 01-.56-.819 2.902 2.902 0 01-.196-1.099c0-.933.224-1.615.672-2.044.448-.43 1.13-.644 2.044-.644.243 0 .488.026.735.077.247.051.478.114.693.189v1.33h1.008v-1.946l.014-.042a6.232 6.232 0 00-1.099-.406c-.434-.121-.996-.182-1.687-.182-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .55.08 1.052.238 1.505.159.453.392.84.7 1.162.308.322.693.57 1.155.742.462.173.996.259 1.603.259zm8.722.014c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V26.8h-3.738v-1.694l-1.092.308V26.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161zm7.966-8.302c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm2.982 8.12v-.938h-2.38V26.8h-3.556v.938h2.436v5.124h-2.436v.938h5.936zm5.194.168a3.82 3.82 0 001.449-.259 2.907 2.907 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm6.58.77v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V33.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078z"/><path id="(){" fill="#7E7C7B" d="M97.272 36.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707zm4.704 0a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm21.728.252v-.98h-1.526c-.57 0-.975-.135-1.218-.406-.243-.27-.364-.69-.364-1.26V33.03a2.25 2.25 0 00-.217-.994 3.347 3.347 0 00-.511-.777 2.744 2.744 0 00-.602-.518c-.205-.126-.369-.198-.49-.217v-.084c.13-.019.299-.086.504-.203a2.7 2.7 0 001.106-1.225c.14-.299.21-.64.21-1.022v-1.358c0-.55.114-.966.343-1.246.229-.28.637-.42 1.225-.42h1.54v-.98h-2.03c-.317 0-.609.058-.875.175a2.053 2.053 0 00-.686.483 2.22 2.22 0 00-.448.728c-.107.28-.161.588-.161.924v1.428c0 .43-.068.789-.203 1.078-.135.29-.303.525-.504.707a1.87 1.87 0 01-.644.392c-.229.08-.432.119-.609.119V31c.177 0 .38.047.609.14.229.093.443.238.644.434.2.196.369.441.504.735s.203.642.203 1.043v1.372c0 .756.198 1.328.595 1.715.397.387.922.581 1.575.581h2.03z"/><path id="alert" fill="#181717" d="M136.514 33.926a2.749 2.749 0 001.589-.469c.182-.126.329-.254.441-.385.112-.13.196-.238.252-.322h.07l.126 1.05h1.862v-.91h-.938c-.056-.42-.084-.863-.084-1.33.01-.401.023-.803.042-1.204.019-.401.028-.78.028-1.134 0-.308-.03-.614-.091-.917a2.049 2.049 0 00-.35-.812 1.893 1.893 0 00-.714-.581c-.303-.15-.693-.224-1.169-.224-.485 0-.98.051-1.484.154-.504.103-.957.27-1.358.504l.35.812c.299-.121.64-.24 1.022-.357a4.106 4.106 0 011.204-.175c.308 0 .567.035.777.105.21.07.373.189.49.357.117.168.191.394.224.679.033.285.026.642-.021 1.071a9.544 9.544 0 00-1.869-.112c-.555.028-1.029.126-1.421.294-.392.168-.693.41-.903.728-.21.317-.315.714-.315 1.19 0 .616.2 1.101.602 1.456.401.355.947.532 1.638.532zm.28-.938c-.495 0-.847-.114-1.057-.343a1.152 1.152 0 01-.315-.805c0-.299.084-.541.252-.728.168-.187.401-.324.7-.413a3.763 3.763 0 011.064-.133c.41 0 .859.042 1.344.126v.98a2.133 2.133 0 01-.672.868 2.22 2.22 0 01-.588.322 2.057 2.057 0 01-.728.126zm9.884.994c.159 0 .331-.019.518-.056a3.61 3.61 0 001.106-.42c.177-.103.34-.22.49-.35l-.504-.77c-.29.224-.553.383-.791.476-.238.093-.488.14-.749.14-.373 0-.656-.117-.847-.35-.191-.233-.287-.635-.287-1.204V24h-2.324v.938h1.232v6.51c0 .476.047.875.14 1.197.093.322.231.581.413.777.182.196.408.338.679.427.27.089.579.133.924.133zm7.672-.014c.299 0 .595-.026.889-.077.294-.051.576-.124.847-.217.27-.093.52-.205.749-.336.229-.13.427-.27.595-.42l-.434-.798a2.585 2.585 0 01-.399.28 3.72 3.72 0 01-.595.28c-.224.084-.46.156-.707.217-.247.06-.497.091-.749.091-.756 0-1.36-.208-1.813-.623-.453-.415-.679-1.043-.679-1.883h5.418c.047-.793-.01-1.44-.168-1.939-.159-.5-.385-.891-.679-1.176a2.295 2.295 0 00-1.015-.581 4.466 4.466 0 00-1.162-.154c-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .56.075 1.064.224 1.512.15.448.369.833.658 1.155.29.322.649.57 1.078.742.43.173.924.259 1.484.259zm2.1-4.466h-4.354c.037-.364.124-.67.259-.917s.308-.448.518-.602c.21-.154.453-.264.728-.329a3.78 3.78 0 01.875-.098c.57 0 1.029.17 1.379.511.35.34.548.819.595 1.435zm8.554 4.298v-.938h-2.688v-4.13c.056-.084.145-.187.266-.308.121-.121.268-.243.441-.364s.376-.224.609-.308c.233-.084.495-.126.784-.126.308 0 .52.14.637.42.117.28.175.719.175 1.316l.98-.014c0-.42-.019-.796-.056-1.127a2.771 2.771 0 00-.217-.847 1.206 1.206 0 00-.448-.532c-.191-.121-.446-.182-.763-.182-.541 0-1.008.103-1.4.308a5.046 5.046 0 00-1.064.742h-.07l-.224-.91h-2.618v.938h1.89v5.124h-1.89v.938h5.656zm6.678.182c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V26.8h-3.738v-1.694l-1.092.308V26.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161z"/><path id="(" fill="#7E7C7B" d="M181.272 36.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707z"/><path id="j" fill="#181717" d="M188.636 25.68c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm-1.904 11.088c.85 0 1.514-.231 1.995-.693.48-.462.721-1.164.721-2.107V26.8h-4.508v.938h3.388v6.02c0 .616-.119 1.099-.357 1.449-.238.35-.637.525-1.197.525-.345 0-.679-.07-1.001-.21a4.362 4.362 0 01-.861-.49l-.462.91a4.501 4.501 0 00.882.497c.205.089.427.166.665.231.238.065.483.098.735.098z"/><path id=");}," fill="#7E7C7B" d="M194.376 36.768a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm10.304-8.302c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm-.756 7.42c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546zm16.45 1.134c.653 0 1.178-.194 1.575-.581.397-.387.595-.959.595-1.715v-1.372c0-.401.068-.749.203-1.043a2.44 2.44 0 01.504-.735c.2-.196.415-.34.644-.434.229-.093.432-.14.609-.14v-.98c-.177 0-.38-.04-.609-.119a1.87 1.87 0 01-.644-.392 2.193 2.193 0 01-.504-.707c-.135-.29-.203-.649-.203-1.078v-1.428c0-.336-.054-.644-.161-.924a2.22 2.22 0 00-.448-.728 2.053 2.053 0 00-.686-.483 2.155 2.155 0 00-.875-.175h-2.03v.98h1.54c.588 0 .996.14 1.225.42.229.28.343.695.343 1.246v1.358c0 .383.07.723.21 1.022a2.7 2.7 0 001.106 1.225c.205.117.373.184.504.203v.084c-.13.019-.299.091-.504.217a2.744 2.744 0 00-.602.518c-.196.22-.364.478-.504.777-.14.299-.21.63-.21.994v1.344c0 .57-.121.99-.364 1.26s-.649.406-1.218.406h-1.526v.98h2.03zm8.75-1.134c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546z"/><path id="function" fill="#1C85B5" d="M22.512 57.8v-.938h-2.898v-5.124h2.898V50.8h-2.898c0-.69.112-1.188.336-1.491.224-.303.644-.455 1.26-.455.187 0 .392.016.616.049.224.033.457.1.7.203l.252-.91a3.603 3.603 0 00-.784-.224 5.473 5.473 0 00-.826-.056c-.85 0-1.503.196-1.96.588-.457.392-.686 1.027-.686 1.904v.392h-1.666v.938h1.666v5.124h-1.666v.938h5.656zm5.222.168c.57 0 1.055-.126 1.456-.378s.723-.579.966-.98h.056l.042 1.19h1.862v-.91h-.938a9.956 9.956 0 01-.084-1.316V50.8h-1.988v.938h.896v3.626c-.187.476-.46.873-.819 1.19-.36.317-.782.476-1.267.476-.336 0-.604-.054-.805-.161a1.046 1.046 0 01-.455-.476 2.317 2.317 0 01-.203-.77 9.693 9.693 0 01-.049-1.029V50.8h-2.03v.938h.938v3.08c0 .607.047 1.113.14 1.519.093.406.238.73.434.973s.446.413.749.511c.303.098.67.147 1.099.147zm7.07-.168v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V57.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078zm10.556.168c.364 0 .702-.033 1.015-.098a4.708 4.708 0 002.023-.952l-.49-.812c-.28.243-.628.439-1.043.588-.415.15-.842.224-1.281.224-.41 0-.786-.06-1.127-.182a2.517 2.517 0 01-.875-.518 2.248 2.248 0 01-.56-.819 2.902 2.902 0 01-.196-1.099c0-.933.224-1.615.672-2.044.448-.43 1.13-.644 2.044-.644.243 0 .488.026.735.077.247.051.478.114.693.189v1.33h1.008v-1.946l.014-.042a6.232 6.232 0 00-1.099-.406c-.434-.121-.996-.182-1.687-.182-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .55.08 1.052.238 1.505.159.453.392.84.7 1.162.308.322.693.57 1.155.742.462.173.996.259 1.603.259zm8.722.014c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V50.8h-3.738v-1.694l-1.092.308V50.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161zm7.966-8.302c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm2.982 8.12v-.938h-2.38V50.8h-3.556v.938h2.436v5.124h-2.436v.938h5.936zm5.194.168a3.82 3.82 0 001.449-.259 2.907 2.907 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm6.58.77v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V57.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078z"/><path id="(){" fill="#7E7C7B" d="M97.272 60.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707zm4.704 0a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm21.728.252v-.98h-1.526c-.57 0-.975-.135-1.218-.406-.243-.27-.364-.69-.364-1.26V57.03a2.25 2.25 0 00-.217-.994 3.347 3.347 0 00-.511-.777 2.744 2.744 0 00-.602-.518c-.205-.126-.369-.198-.49-.217v-.084c.13-.019.299-.086.504-.203a2.7 2.7 0 001.106-1.225c.14-.299.21-.64.21-1.022v-1.358c0-.55.114-.966.343-1.246.229-.28.637-.42 1.225-.42h1.54v-.98h-2.03c-.317 0-.609.058-.875.175a2.053 2.053 0 00-.686.483 2.22 2.22 0 00-.448.728c-.107.28-.161.588-.161.924v1.428c0 .43-.068.789-.203 1.078-.135.29-.303.525-.504.707a1.87 1.87 0 01-.644.392c-.229.08-.432.119-.609.119V55c.177 0 .38.047.609.14.229.093.443.238.644.434.2.196.369.441.504.735s.203.642.203 1.043v1.372c0 .756.198 1.328.595 1.715.397.387.922.581 1.575.581h2.03z"/><path id="alert" fill="#181717" d="M136.514 57.926a2.749 2.749 0 001.589-.469c.182-.126.329-.254.441-.385.112-.13.196-.238.252-.322h.07l.126 1.05h1.862v-.91h-.938c-.056-.42-.084-.863-.084-1.33.01-.401.023-.803.042-1.204.019-.401.028-.78.028-1.134 0-.308-.03-.614-.091-.917a2.049 2.049 0 00-.35-.812 1.893 1.893 0 00-.714-.581c-.303-.15-.693-.224-1.169-.224-.485 0-.98.051-1.484.154-.504.103-.957.27-1.358.504l.35.812c.299-.121.64-.24 1.022-.357a4.106 4.106 0 011.204-.175c.308 0 .567.035.777.105.21.07.373.189.49.357.117.168.191.394.224.679.033.285.026.642-.021 1.071a9.544 9.544 0 00-1.869-.112c-.555.028-1.029.126-1.421.294-.392.168-.693.41-.903.728-.21.317-.315.714-.315 1.19 0 .616.2 1.101.602 1.456.401.355.947.532 1.638.532zm.28-.938c-.495 0-.847-.114-1.057-.343a1.152 1.152 0 01-.315-.805c0-.299.084-.541.252-.728.168-.187.401-.324.7-.413a3.763 3.763 0 011.064-.133c.41 0 .859.042 1.344.126v.98a2.133 2.133 0 01-.672.868 2.22 2.22 0 01-.588.322 2.057 2.057 0 01-.728.126zm9.884.994c.159 0 .331-.019.518-.056a3.61 3.61 0 001.106-.42c.177-.103.34-.22.49-.35l-.504-.77c-.29.224-.553.383-.791.476-.238.093-.488.14-.749.14-.373 0-.656-.117-.847-.35-.191-.233-.287-.635-.287-1.204V48h-2.324v.938h1.232v6.51c0 .476.047.875.14 1.197.093.322.231.581.413.777.182.196.408.338.679.427.27.089.579.133.924.133zm7.672-.014c.299 0 .595-.026.889-.077.294-.051.576-.124.847-.217.27-.093.52-.205.749-.336.229-.13.427-.27.595-.42l-.434-.798a2.585 2.585 0 01-.399.28 3.72 3.72 0 01-.595.28c-.224.084-.46.156-.707.217-.247.06-.497.091-.749.091-.756 0-1.36-.208-1.813-.623-.453-.415-.679-1.043-.679-1.883h5.418c.047-.793-.01-1.44-.168-1.939-.159-.5-.385-.891-.679-1.176a2.295 2.295 0 00-1.015-.581 4.466 4.466 0 00-1.162-.154c-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .56.075 1.064.224 1.512.15.448.369.833.658 1.155.29.322.649.57 1.078.742.43.173.924.259 1.484.259zm2.1-4.466h-4.354c.037-.364.124-.67.259-.917s.308-.448.518-.602c.21-.154.453-.264.728-.329a3.78 3.78 0 01.875-.098c.57 0 1.029.17 1.379.511.35.34.548.819.595 1.435zm8.554 4.298v-.938h-2.688v-4.13c.056-.084.145-.187.266-.308.121-.121.268-.243.441-.364s.376-.224.609-.308c.233-.084.495-.126.784-.126.308 0 .52.14.637.42.117.28.175.719.175 1.316l.98-.014c0-.42-.019-.796-.056-1.127a2.771 2.771 0 00-.217-.847 1.206 1.206 0 00-.448-.532c-.191-.121-.446-.182-.763-.182-.541 0-1.008.103-1.4.308a5.046 5.046 0 00-1.064.742h-.07l-.224-.91h-2.618v.938h1.89v5.124h-1.89v.938h5.656zm6.678.182c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V50.8h-3.738v-1.694l-1.092.308V50.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161z"/><path id="(" fill="#7E7C7B" d="M181.272 60.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707z"/><path id="j" fill="#181717" d="M188.636 49.68c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm-1.904 11.088c.85 0 1.514-.231 1.995-.693.48-.462.721-1.164.721-2.107V50.8h-4.508v.938h3.388v6.02c0 .616-.119 1.099-.357 1.449-.238.35-.637.525-1.197.525-.345 0-.679-.07-1.001-.21a4.362 4.362 0 01-.861-.49l-.462.91a4.501 4.501 0 00.882.497c.205.089.427.166.665.231.238.065.483.098.735.098z"/><path id=");}," fill="#7E7C7B" d="M194.376 60.768a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm10.304-8.302c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm-.756 7.42c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546zm16.45 1.134c.653 0 1.178-.194 1.575-.581.397-.387.595-.959.595-1.715v-1.372c0-.401.068-.749.203-1.043a2.44 2.44 0 01.504-.735c.2-.196.415-.34.644-.434.229-.093.432-.14.609-.14v-.98c-.177 0-.38-.04-.609-.119a1.87 1.87 0 01-.644-.392 2.193 2.193 0 01-.504-.707c-.135-.29-.203-.649-.203-1.078v-1.428c0-.336-.054-.644-.161-.924a2.22 2.22 0 00-.448-.728 2.053 2.053 0 00-.686-.483 2.155 2.155 0 00-.875-.175h-2.03v.98h1.54c.588 0 .996.14 1.225.42.229.28.343.695.343 1.246v1.358c0 .383.07.723.21 1.022a2.7 2.7 0 001.106 1.225c.205.117.373.184.504.203v.084c-.13.019-.299.091-.504.217a2.744 2.744 0 00-.602.518c-.196.22-.364.478-.504.777-.14.299-.21.63-.21.994v1.344c0 .57-.121.99-.364 1.26s-.649.406-1.218.406h-1.526v.98h2.03zm8.75-1.134c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546z"/><path id="function" fill="#1C85B5" d="M22.512 81.8v-.938h-2.898v-5.124h2.898V74.8h-2.898c0-.69.112-1.188.336-1.491.224-.303.644-.455 1.26-.455.187 0 .392.016.616.049.224.033.457.1.7.203l.252-.91a3.603 3.603 0 00-.784-.224 5.473 5.473 0 00-.826-.056c-.85 0-1.503.196-1.96.588-.457.392-.686 1.027-.686 1.904v.392h-1.666v.938h1.666v5.124h-1.666v.938h5.656zm5.222.168c.57 0 1.055-.126 1.456-.378s.723-.579.966-.98h.056l.042 1.19h1.862v-.91h-.938a9.956 9.956 0 01-.084-1.316V74.8h-1.988v.938h.896v3.626c-.187.476-.46.873-.819 1.19-.36.317-.782.476-1.267.476-.336 0-.604-.054-.805-.161a1.046 1.046 0 01-.455-.476 2.317 2.317 0 01-.203-.77 9.693 9.693 0 01-.049-1.029V74.8h-2.03v.938h.938v3.08c0 .607.047 1.113.14 1.519.093.406.238.73.434.973s.446.413.749.511c.303.098.67.147 1.099.147zm7.07-.168v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V81.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078zm10.556.168c.364 0 .702-.033 1.015-.098a4.708 4.708 0 002.023-.952l-.49-.812c-.28.243-.628.439-1.043.588-.415.15-.842.224-1.281.224-.41 0-.786-.06-1.127-.182a2.517 2.517 0 01-.875-.518 2.248 2.248 0 01-.56-.819 2.902 2.902 0 01-.196-1.099c0-.933.224-1.615.672-2.044.448-.43 1.13-.644 2.044-.644.243 0 .488.026.735.077.247.051.478.114.693.189v1.33h1.008v-1.946l.014-.042a6.232 6.232 0 00-1.099-.406c-.434-.121-.996-.182-1.687-.182-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .55.08 1.052.238 1.505.159.453.392.84.7 1.162.308.322.693.57 1.155.742.462.173.996.259 1.603.259zm8.722.014c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V74.8h-3.738v-1.694l-1.092.308V74.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161zm7.966-8.302c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm2.982 8.12v-.938h-2.38V74.8h-3.556v.938h2.436v5.124h-2.436v.938h5.936zm5.194.168a3.82 3.82 0 001.449-.259 2.907 2.907 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm6.58.77v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75V81.8h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.159 12.159 0 01.056.847c.005.135.007.245.007.329v4.55h1.078z"/><path id="(){" fill="#7E7C7B" d="M97.272 84.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707zm4.704 0a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm21.728.252v-.98h-1.526c-.57 0-.975-.135-1.218-.406-.243-.27-.364-.69-.364-1.26V81.03a2.25 2.25 0 00-.217-.994 3.347 3.347 0 00-.511-.777 2.744 2.744 0 00-.602-.518c-.205-.126-.369-.198-.49-.217v-.084c.13-.019.299-.086.504-.203a2.7 2.7 0 001.106-1.225c.14-.299.21-.64.21-1.022v-1.358c0-.55.114-.966.343-1.246.229-.28.637-.42 1.225-.42h1.54v-.98h-2.03c-.317 0-.609.058-.875.175a2.053 2.053 0 00-.686.483 2.22 2.22 0 00-.448.728c-.107.28-.161.588-.161.924v1.428c0 .43-.068.789-.203 1.078-.135.29-.303.525-.504.707a1.87 1.87 0 01-.644.392c-.229.08-.432.119-.609.119V79c.177 0 .38.047.609.14.229.093.443.238.644.434.2.196.369.441.504.735s.203.642.203 1.043v1.372c0 .756.198 1.328.595 1.715.397.387.922.581 1.575.581h2.03z"/><path id="alert" fill="#181717" d="M136.514 81.926a2.749 2.749 0 001.589-.469c.182-.126.329-.254.441-.385.112-.13.196-.238.252-.322h.07l.126 1.05h1.862v-.91h-.938c-.056-.42-.084-.863-.084-1.33.01-.401.023-.803.042-1.204.019-.401.028-.78.028-1.134 0-.308-.03-.614-.091-.917a2.049 2.049 0 00-.35-.812 1.893 1.893 0 00-.714-.581c-.303-.15-.693-.224-1.169-.224-.485 0-.98.051-1.484.154-.504.103-.957.27-1.358.504l.35.812c.299-.121.64-.24 1.022-.357a4.106 4.106 0 011.204-.175c.308 0 .567.035.777.105.21.07.373.189.49.357.117.168.191.394.224.679.033.285.026.642-.021 1.071a9.544 9.544 0 00-1.869-.112c-.555.028-1.029.126-1.421.294-.392.168-.693.41-.903.728-.21.317-.315.714-.315 1.19 0 .616.2 1.101.602 1.456.401.355.947.532 1.638.532zm.28-.938c-.495 0-.847-.114-1.057-.343a1.152 1.152 0 01-.315-.805c0-.299.084-.541.252-.728.168-.187.401-.324.7-.413a3.763 3.763 0 011.064-.133c.41 0 .859.042 1.344.126v.98a2.133 2.133 0 01-.672.868 2.22 2.22 0 01-.588.322 2.057 2.057 0 01-.728.126zm9.884.994c.159 0 .331-.019.518-.056a3.61 3.61 0 001.106-.42c.177-.103.34-.22.49-.35l-.504-.77c-.29.224-.553.383-.791.476-.238.093-.488.14-.749.14-.373 0-.656-.117-.847-.35-.191-.233-.287-.635-.287-1.204V72h-2.324v.938h1.232v6.51c0 .476.047.875.14 1.197.093.322.231.581.413.777.182.196.408.338.679.427.27.089.579.133.924.133zm7.672-.014c.299 0 .595-.026.889-.077.294-.051.576-.124.847-.217.27-.093.52-.205.749-.336.229-.13.427-.27.595-.42l-.434-.798a2.585 2.585 0 01-.399.28 3.72 3.72 0 01-.595.28c-.224.084-.46.156-.707.217-.247.06-.497.091-.749.091-.756 0-1.36-.208-1.813-.623-.453-.415-.679-1.043-.679-1.883h5.418c.047-.793-.01-1.44-.168-1.939-.159-.5-.385-.891-.679-1.176a2.295 2.295 0 00-1.015-.581 4.466 4.466 0 00-1.162-.154c-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .56.075 1.064.224 1.512.15.448.369.833.658 1.155.29.322.649.57 1.078.742.43.173.924.259 1.484.259zm2.1-4.466h-4.354c.037-.364.124-.67.259-.917s.308-.448.518-.602c.21-.154.453-.264.728-.329a3.78 3.78 0 01.875-.098c.57 0 1.029.17 1.379.511.35.34.548.819.595 1.435zm8.554 4.298v-.938h-2.688v-4.13c.056-.084.145-.187.266-.308.121-.121.268-.243.441-.364s.376-.224.609-.308c.233-.084.495-.126.784-.126.308 0 .52.14.637.42.117.28.175.719.175 1.316l.98-.014c0-.42-.019-.796-.056-1.127a2.771 2.771 0 00-.217-.847 1.206 1.206 0 00-.448-.532c-.191-.121-.446-.182-.763-.182-.541 0-1.008.103-1.4.308a5.046 5.046 0 00-1.064.742h-.07l-.224-.91h-2.618v.938h1.89v5.124h-1.89v.938h5.656zm6.678.182c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738V74.8h-3.738v-1.694l-1.092.308V74.8h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161z"/><path id="(" fill="#7E7C7B" d="M181.272 84.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707z"/><path id="j" fill="#181717" d="M188.636 73.68c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm-1.904 11.088c.85 0 1.514-.231 1.995-.693.48-.462.721-1.164.721-2.107V74.8h-4.508v.938h3.388v6.02c0 .616-.119 1.099-.357 1.449-.238.35-.637.525-1.197.525-.345 0-.679-.07-1.001-.21a4.362 4.362 0 01-.861-.49l-.462.91a4.501 4.501 0 00.882.497c.205.089.427.166.665.231.238.065.483.098.735.098z"/><path id=");}," fill="#7E7C7B" d="M194.376 84.768a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm10.304-8.302c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm-.756 7.42c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546zm16.45 1.134c.653 0 1.178-.194 1.575-.581.397-.387.595-.959.595-1.715v-1.372c0-.401.068-.749.203-1.043a2.44 2.44 0 01.504-.735c.2-.196.415-.34.644-.434.229-.093.432-.14.609-.14v-.98c-.177 0-.38-.04-.609-.119a1.87 1.87 0 01-.644-.392 2.193 2.193 0 01-.504-.707c-.135-.29-.203-.649-.203-1.078v-1.428c0-.336-.054-.644-.161-.924a2.22 2.22 0 00-.448-.728 2.053 2.053 0 00-.686-.483 2.155 2.155 0 00-.875-.175h-2.03v.98h1.54c.588 0 .996.14 1.225.42.229.28.343.695.343 1.246v1.358c0 .383.07.723.21 1.022a2.7 2.7 0 001.106 1.225c.205.117.373.184.504.203v.084c-.13.019-.299.091-.504.217a2.744 2.744 0 00-.602.518c-.196.22-.364.478-.504.777-.14.299-.21.63-.21.994v1.344c0 .57-.121.99-.364 1.26s-.649.406-1.218.406h-1.526v.98h2.03zm8.75-1.134c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546z"/><path id="..." fill="#181717" d="M19.81 105.968c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm8.4 0c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm8.4 0c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266z"/><path id="function" fill="#1C85B5" d="M22.512 129.8v-.938h-2.898v-5.124h2.898v-.938h-2.898c0-.69.112-1.188.336-1.491.224-.303.644-.455 1.26-.455.187 0 .392.016.616.049.224.033.457.1.7.203l.252-.91a3.603 3.603 0 00-.784-.224 5.473 5.473 0 00-.826-.056c-.85 0-1.503.196-1.96.588-.457.392-.686 1.027-.686 1.904v.392h-1.666v.938h1.666v5.124h-1.666v.938h5.656zm5.222.168c.57 0 1.055-.126 1.456-.378s.723-.579.966-.98h.056l.042 1.19h1.862v-.91h-.938a9.956 9.956 0 01-.084-1.316V122.8h-1.988v.938h.896v3.626c-.187.476-.46.873-.819 1.19-.36.317-.782.476-1.267.476-.336 0-.604-.054-.805-.161a1.046 1.046 0 01-.455-.476 2.317 2.317 0 01-.203-.77 9.693 9.693 0 01-.049-1.029V122.8h-2.03v.938h.938v3.08c0 .607.047 1.113.14 1.519.093.406.238.73.434.973s.446.413.749.511c.303.098.67.147 1.099.147zm7.07-.168v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75v3.808h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.158 12.158 0 01.056.847c.005.135.007.245.007.329v4.55h1.078zm10.556.168c.364 0 .702-.033 1.015-.098a4.708 4.708 0 002.023-.952l-.49-.812c-.28.243-.628.439-1.043.588-.415.15-.842.224-1.281.224-.41 0-.786-.06-1.127-.182a2.517 2.517 0 01-.875-.518 2.248 2.248 0 01-.56-.819 2.902 2.902 0 01-.196-1.099c0-.933.224-1.615.672-2.044.448-.43 1.13-.644 2.044-.644.243 0 .488.026.735.077.247.051.478.114.693.189v1.33h1.008v-1.946l.014-.042a6.232 6.232 0 00-1.099-.406c-.434-.121-.996-.182-1.687-.182-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .55.08 1.052.238 1.505.159.453.392.84.7 1.162.308.322.693.57 1.155.742.462.173.996.259 1.603.259zm8.722.014c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738v-.938h-3.738v-1.694l-1.092.308v1.386h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161zm7.966-8.302c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm2.982 8.12v-.938h-2.38V122.8h-3.556v.938h2.436v5.124h-2.436v.938h5.936zm5.194.168a3.82 3.82 0 001.449-.259 2.908 2.908 0 001.071-.735c.29-.317.509-.702.658-1.155.15-.453.224-.959.224-1.519 0-.513-.068-.994-.203-1.442a3.168 3.168 0 00-.623-1.162 2.981 2.981 0 00-1.057-.777c-.425-.191-.931-.287-1.519-.287-1.092 0-1.932.329-2.52.987-.588.658-.882 1.552-.882 2.681 0 .513.068.994.203 1.442.135.448.343.835.623 1.162.28.327.635.586 1.064.777.43.191.933.287 1.512.287zm0-.938c-.41 0-.758-.08-1.043-.238a2.001 2.001 0 01-.693-.63 2.683 2.683 0 01-.385-.882 4.14 4.14 0 01-.119-.98c0-.905.187-1.587.56-2.044.373-.457.933-.686 1.68-.686.41 0 .758.08 1.043.238.285.159.518.369.7.63.182.261.313.553.392.875.08.322.119.651.119.987 0 .896-.189 1.575-.567 2.037-.378.462-.94.693-1.687.693zm6.58.77v-4.396c.065-.233.168-.455.308-.665.14-.21.303-.394.49-.553a2.44 2.44 0 01.623-.385c.229-.098.464-.147.707-.147.663 0 1.12.196 1.372.588.252.392.378.975.378 1.75v3.808h1.092v-4.018c0-.625-.056-1.141-.168-1.547-.112-.406-.278-.728-.497-.966a1.738 1.738 0 00-.826-.497 4.219 4.219 0 00-1.141-.14 2.856 2.856 0 00-2.023.833c-.154.154-.278.31-.371.469h-.07l-.126-1.134h-1.876v.91h.938c.019.103.035.224.049.364a12.158 12.158 0 01.056.847c.005.135.007.245.007.329v4.55h1.078z"/><path id="(){" fill="#7E7C7B" d="M97.272 132.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707zm4.704 0a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm21.728.252v-.98h-1.526c-.57 0-.975-.135-1.218-.406-.243-.27-.364-.69-.364-1.26v-1.344a2.25 2.25 0 00-.217-.994 3.347 3.347 0 00-.511-.777 2.744 2.744 0 00-.602-.518c-.205-.126-.369-.198-.49-.217v-.084c.13-.019.299-.086.504-.203a2.7 2.7 0 001.106-1.225c.14-.299.21-.64.21-1.022v-1.358c0-.55.114-.966.343-1.246.229-.28.637-.42 1.225-.42h1.54v-.98h-2.03c-.317 0-.609.058-.875.175a2.053 2.053 0 00-.686.483 2.22 2.22 0 00-.448.728c-.107.28-.161.588-.161.924v1.428c0 .43-.068.789-.203 1.078-.135.29-.303.525-.504.707a1.87 1.87 0 01-.644.392c-.229.08-.432.119-.609.119v.98c.177 0 .38.047.609.14.229.093.443.238.644.434.2.196.369.441.504.735s.203.642.203 1.043v1.372c0 .756.198 1.328.595 1.715.397.387.922.581 1.575.581h2.03z"/><path id="alert" fill="#181717" d="M136.514 129.926a2.749 2.749 0 001.589-.469c.182-.126.329-.254.441-.385.112-.13.196-.238.252-.322h.07l.126 1.05h1.862v-.91h-.938c-.056-.42-.084-.863-.084-1.33.01-.401.023-.803.042-1.204.019-.401.028-.78.028-1.134 0-.308-.03-.614-.091-.917a2.049 2.049 0 00-.35-.812 1.893 1.893 0 00-.714-.581c-.303-.15-.693-.224-1.169-.224-.485 0-.98.051-1.484.154-.504.103-.957.27-1.358.504l.35.812c.299-.121.64-.24 1.022-.357a4.106 4.106 0 011.204-.175c.308 0 .567.035.777.105.21.07.373.189.49.357.117.168.191.394.224.679.033.285.026.642-.021 1.071a9.544 9.544 0 00-1.869-.112c-.555.028-1.029.126-1.421.294-.392.168-.693.41-.903.728-.21.317-.315.714-.315 1.19 0 .616.2 1.101.602 1.456.401.355.947.532 1.638.532zm.28-.938c-.495 0-.847-.114-1.057-.343a1.152 1.152 0 01-.315-.805c0-.299.084-.541.252-.728.168-.187.401-.324.7-.413a3.763 3.763 0 011.064-.133c.41 0 .859.042 1.344.126v.98a2.133 2.133 0 01-.672.868 2.22 2.22 0 01-.588.322 2.057 2.057 0 01-.728.126zm9.884.994c.159 0 .331-.019.518-.056a3.61 3.61 0 001.106-.42c.177-.103.34-.22.49-.35l-.504-.77c-.29.224-.553.383-.791.476-.238.093-.488.14-.749.14-.373 0-.656-.117-.847-.35-.191-.233-.287-.635-.287-1.204V120h-2.324v.938h1.232v6.51c0 .476.047.875.14 1.197.093.322.231.581.413.777.182.196.408.338.679.427.27.089.579.133.924.133zm7.672-.014c.299 0 .595-.026.889-.077.294-.051.576-.124.847-.217.27-.093.52-.205.749-.336.229-.13.427-.27.595-.42l-.434-.798a2.585 2.585 0 01-.399.28 3.72 3.72 0 01-.595.28c-.224.084-.46.156-.707.217-.247.06-.497.091-.749.091-.756 0-1.36-.208-1.813-.623-.453-.415-.679-1.043-.679-1.883h5.418c.047-.793-.01-1.44-.168-1.939-.159-.5-.385-.891-.679-1.176a2.295 2.295 0 00-1.015-.581 4.466 4.466 0 00-1.162-.154c-.541 0-1.031.082-1.47.245a3.08 3.08 0 00-1.12.707c-.308.308-.544.69-.707 1.148-.163.457-.245.98-.245 1.568 0 .56.075 1.064.224 1.512.15.448.369.833.658 1.155.29.322.649.57 1.078.742.43.173.924.259 1.484.259zm2.1-4.466h-4.354c.037-.364.124-.67.259-.917s.308-.448.518-.602c.21-.154.453-.264.728-.329a3.78 3.78 0 01.875-.098c.57 0 1.029.17 1.379.511.35.34.548.819.595 1.435zm8.554 4.298v-.938h-2.688v-4.13c.056-.084.145-.187.266-.308.121-.121.268-.243.441-.364s.376-.224.609-.308c.233-.084.495-.126.784-.126.308 0 .52.14.637.42.117.28.175.719.175 1.316l.98-.014c0-.42-.019-.796-.056-1.127a2.771 2.771 0 00-.217-.847 1.206 1.206 0 00-.448-.532c-.191-.121-.446-.182-.763-.182-.541 0-1.008.103-1.4.308a5.046 5.046 0 00-1.064.742h-.07l-.224-.91h-2.618v.938h1.89v5.124h-1.89v.938h5.656zm6.678.182c.476 0 .94-.08 1.393-.238a3.938 3.938 0 001.169-.63l-.364-.826c-.252.187-.534.36-.847.518-.313.159-.679.238-1.099.238-.616 0-1.09-.163-1.421-.49-.331-.327-.497-.83-.497-1.512v-3.304h3.738v-.938h-3.738v-1.694l-1.092.308v1.386h-1.666v.938h1.666v3.472c0 .485.075.903.224 1.253.15.35.35.637.602.861.252.224.544.39.875.497a3.41 3.41 0 001.057.161z"/><path id="(" fill="#7E7C7B" d="M181.272 132.768l.35-.966a4.965 4.965 0 01-1.337-.497 4.15 4.15 0 01-1.204-.994c-.36-.425-.649-.957-.868-1.596-.22-.64-.329-1.412-.329-2.317 0-.896.091-1.659.273-2.289.182-.63.439-1.157.77-1.582a3.928 3.928 0 011.183-1.022 7.086 7.086 0 011.512-.623l-.35-.882a6.005 6.005 0 00-1.729.665 4.984 4.984 0 00-1.435 1.246c-.41.518-.737 1.148-.98 1.89s-.364 1.608-.364 2.597c0 .999.119 1.862.357 2.59.238.728.56 1.344.966 1.848.406.504.882.912 1.428 1.225a6.964 6.964 0 001.757.707z"/><path id="j" fill="#181717" d="M188.636 121.68c.27 0 .497-.084.679-.252a.806.806 0 00.273-.616.889.889 0 00-.273-.651.922.922 0 00-.679-.273.838.838 0 00-.644.273.927.927 0 00-.252.651c0 .243.084.448.252.616a.872.872 0 00.644.252zm-1.904 11.088c.85 0 1.514-.231 1.995-.693.48-.462.721-1.164.721-2.107V122.8h-4.508v.938h3.388v6.02c0 .616-.119 1.099-.357 1.449-.238.35-.637.525-1.197.525-.345 0-.679-.07-1.001-.21a4.362 4.362 0 01-.861-.49l-.462.91a4.501 4.501 0 00.882.497c.205.089.427.166.665.231.238.065.483.098.735.098z"/><path id=");}];" fill="#7E7C7B" d="M194.376 132.768a6.964 6.964 0 001.757-.707 5.036 5.036 0 001.428-1.225c.406-.504.728-1.12.966-1.848s.357-1.591.357-2.59c0-.99-.121-1.855-.364-2.597s-.57-1.372-.98-1.89a4.984 4.984 0 00-1.435-1.246 6.005 6.005 0 00-1.729-.665l-.35.882c.55.159 1.055.366 1.512.623.457.257.852.597 1.183 1.022.331.425.588.952.77 1.582.182.63.273 1.393.273 2.289 0 .905-.11 1.678-.329 2.317-.22.64-.509 1.171-.868 1.596a4.15 4.15 0 01-1.204.994 4.965 4.965 0 01-1.337.497l.35.966zm10.304-8.302c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm-.756 7.42c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546zm16.45 1.134c.653 0 1.178-.194 1.575-.581.397-.387.595-.959.595-1.715v-1.372c0-.401.068-.749.203-1.043a2.44 2.44 0 01.504-.735c.2-.196.415-.34.644-.434.229-.093.432-.14.609-.14v-.98c-.177 0-.38-.04-.609-.119a1.87 1.87 0 01-.644-.392 2.193 2.193 0 01-.504-.707c-.135-.29-.203-.649-.203-1.078v-1.428c0-.336-.054-.644-.161-.924a2.22 2.22 0 00-.448-.728 2.053 2.053 0 00-.686-.483 2.155 2.155 0 00-.875-.175h-2.03v.98h1.54c.588 0 .996.14 1.225.42.229.28.343.695.343 1.246v1.358c0 .383.07.723.21 1.022a2.7 2.7 0 001.106 1.225c.205.117.373.184.504.203v.084c-.13.019-.299.091-.504.217a2.744 2.744 0 00-.602.518c-.196.22-.364.478-.504.777-.14.299-.21.63-.21.994v1.344c0 .57-.121.99-.364 1.26s-.649.406-1.218.406h-1.526v.98h2.03zm-215.236 24V144H.35v.98h3.696v11.06H.35v.98h4.788zm6.342-8.554c.299 0 .532-.089.7-.266a.938.938 0 00.252-.672c0-.299-.084-.53-.252-.693-.168-.163-.401-.245-.7-.245-.28 0-.504.082-.672.245-.168.163-.252.394-.252.693 0 .27.084.495.252.672.168.177.392.266.672.266zm-.756 7.42c.177-.056.371-.142.581-.259a2.42 2.42 0 001.043-1.162 2.36 2.36 0 00.182-.959c0-.448-.098-.796-.294-1.043-.196-.247-.471-.371-.826-.371-.29 0-.525.086-.707.259a.893.893 0 00-.273.679c0 .299.077.525.231.679.154.154.366.231.637.231.075 0 .177-.014.308-.042 0 .345-.121.64-.364.882-.243.243-.532.43-.868.56l.35.546z"/></g><g id="Group-Copy" transform="translate(354 66)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-Copy-2" fill="#C06334" fill-rule="nonzero" transform="translate(242 44)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-3" transform="translate(279 36)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="j:-0" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">j: 0</tspan></text></g><g id="Group-Copy-3" fill="#C06334" fill-rule="nonzero" transform="translate(242 67)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-3-Copy" transform="translate(279 59)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="j:-1" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">j: 1</tspan></text></g><g id="Group-Copy-4" fill="#C06334" fill-rule="nonzero" transform="translate(242 90)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-3-Copy-2" transform="translate(279 82)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="j:-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="14">j: 2</tspan></text></g><g id="Group-Copy-5" fill="#C06334" fill-rule="nonzero" transform="translate(242 139)"><path id="Line" d="M13.5-5.5l14 7-14 7v-6h-14v-2h14v-6z"/></g><g id="Group-3-Copy-3" transform="translate(279 131)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h52v18H1z"/><text id="j:-10" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7.447" y="13.51">j: 10</tspan></text></g><text id="..." fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="403" y="94">...</tspan></text><text id="makeArmy()-LexicalEn" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="403" y="62">makeArmy()</tspan> <tspan x="403" y="77" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text><text id="while-iteration-Lexi" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="278" y="14">while iteration</tspan> <tspan x="278" y="29" font-family="PTMono-Regular, PT Mono" font-weight="normal">LexicalEnvironment</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/8-make-army/lexenv-makearmy.svg b/1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy.svg similarity index 100% rename from 1-js/06-advanced-functions/03-closure/8-make-army/lexenv-makearmy.svg rename to 1-js/06-advanced-functions/03-closure/10-make-army/lexenv-makearmy.svg diff --git a/1-js/06-advanced-functions/03-closure/10-make-army/solution.md b/1-js/06-advanced-functions/03-closure/10-make-army/solution.md new file mode 100644 index 000000000..9d99aa717 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/10-make-army/solution.md @@ -0,0 +1,129 @@ + +Let's examine what exactly happens inside `makeArmy`, and the solution will become obvious. + +1. It creates an empty array `shooters`: + + ```js + let shooters = []; + ``` +2. Fills it with functions via `shooters.push(function)` in the loop. + + Every element is a function, so the resulting array looks like this: + + ```js no-beautify + shooters = [ + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); }, + function () { alert(i); } + ]; + ``` + +3. The array is returned from the function. + + Then, later, the call to any member, e.g. `army[5]()` will get the element `army[5]` from the array (which is a function) and calls it. + + Now why do all such functions show the same value, `10`? + + That's because there's no local variable `i` inside `shooter` functions. When such a function is called, it takes `i` from its outer lexical environment. + + Then, what will be the value of `i`? + + If we look at the source: + + ```js + function makeArmy() { + ... + let i = 0; + while (i < 10) { + let shooter = function() { // shooter function + alert( i ); // should show its number + }; + shooters.push(shooter); // add function to the array + i++; + } + ... + } + ``` + + We can see that all `shooter` functions are created in the lexical environment of `makeArmy()` function. But when `army[5]()` is called, `makeArmy` has already finished its job, and the final value of `i` is `10` (`while` stops at `i=10`). + + As the result, all `shooter` functions get the same value from the outer lexical environment and that is, the last value, `i=10`. + +  + + As you can see above, on each iteration of a `while {...}` block, a new lexical environment is created. So, to fix this, we can copy the value of `i` into a variable within the `while {...}` block, like this: + + ```js run + function makeArmy() { + let shooters = []; + + let i = 0; + while (i < 10) { + *!* + let j = i; + */!* + let shooter = function() { // shooter function + alert( *!*j*/!* ); // should show its number + }; + shooters.push(shooter); + i++; + } + + return shooters; + } + + let army = makeArmy(); + + // Now the code works correctly + army[0](); // 0 + army[5](); // 5 + ``` + + Here `let j = i` declares an "iteration-local" variable `j` and copies `i` into it. Primitives are copied "by value", so we actually get an independent copy of `i`, belonging to the current loop iteration. + + The shooters work correctly, because the value of `i` now lives a little bit closer. Not in `makeArmy()` Lexical Environment, but in the Lexical Environment that corresponds to the current loop iteration: + +  + + Such a problem could also be avoided if we used `for` in the beginning, like this: + + ```js run demo + function makeArmy() { + + let shooters = []; + + *!* + for(let i = 0; i < 10; i++) { + */!* + let shooter = function() { // shooter function + alert( i ); // should show its number + }; + shooters.push(shooter); + } + + return shooters; + } + + let army = makeArmy(); + + army[0](); // 0 + army[5](); // 5 + ``` + + That's essentially the same, because `for` on each iteration generates a new lexical environment, with its own variable `i`. So `shooter` generated in every iteration references its own `i`, from that very iteration. + +  + +Now, as you've put so much effort into reading this, and the final recipe is so simple - just use `for`, you may wonder -- was it worth that? + +Well, if you could easily answer the question, you wouldn't read the solution. So, hopefully this task must have helped you to understand things a bit better. + +Besides, there are indeed cases when one prefers `while` to `for`, and other scenarios, where such problems are real. + diff --git a/1-js/06-advanced-functions/03-closure/10-make-army/task.md b/1-js/06-advanced-functions/03-closure/10-make-army/task.md new file mode 100644 index 000000000..f50c7dc20 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/10-make-army/task.md @@ -0,0 +1,41 @@ +importance: 5 + +--- + +# Army of functions + +The following code creates an array of `shooters`. + +Every function is meant to output its number. But something is wrong... + +```js run +function makeArmy() { + let shooters = []; + + let i = 0; + while (i < 10) { + let shooter = function() { // create a shooter function, + alert( i ); // that should show its number + }; + shooters.push(shooter); // and add it to the array + i++; + } + + // ...and return the array of shooters + return shooters; +} + +let army = makeArmy(); + +*!* +// all shooters show 10 instead of their numbers 0, 1, 2, 3... +army[0](); // 10 from the shooter number 0 +army[1](); // 10 from the shooter number 1 +army[2](); // 10 ...and so on. +*/!* +``` + +Why do all of the shooters show the same value? + +Fix the code so that they work as intended. + diff --git a/1-js/06-advanced-functions/03-closure/2-closure-variable-access/lexenv-nested-work.svg b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/lexenv-nested-work.svg new file mode 100644 index 000000000..8dfd8bd63 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/lexenv-nested-work.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="762" height="225" viewBox="0 0 762 225"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="lexenv-nested-work.svg"><g id="function-makeWorker(" fill-rule="nonzero" transform="translate(10 15)"><path id="function" fill="#1C85B5" d="M1.092 9.912v-.938h1.666V3.85H1.092v-.938h1.666V2.52c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H3.85v5.124h2.898v.938H1.092zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H14.49l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519V3.85H8.61v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.85h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.104c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zM31.206 3.99a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042V5.32h-1.008V3.99zm2.688-1.078h1.666V1.526l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.85h-1.666v-.938zm9.436 7v-.938h2.436V3.85H43.33v-.938h3.556v6.062h2.38v.938H43.33zM45.388.924c0-.252.084-.469.252-.651A.838.838 0 0146.284 0c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.104c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="makeWorker" fill="#181717" d="M79.128 9.912V5.25c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zM87.43 9.1c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM92.596.112h2.268v5.796h.77l2.758-2.996h1.316L96.614 6.23l2.548 2.744h.91v.938H98.63l-2.996-3.164h-.77v3.164h-1.078V1.05h-1.19V.112zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.37 3.43l.07.91h.098l.196-.994 1.036-3.822h.924l.966 3.794.252 1.036h.07l.084-.952 1.036-6.958h1.036l-1.512 9.8h-1.162l-1.134-3.976-.14-.952h-.042l-.154.966-1.134 3.962h-1.176l-1.526-9.8h1.106l1.106 6.986zm6.776-.686c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.364 3.5v-.938h1.89V3.85h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.812-9.8h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078V1.05h-1.19V.112zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.85h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="(){" fill="#7E7C7B" d="M165.508 12.88a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848S161 7.509 161 6.51c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.408c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.464z"/><path id="let" fill="#1C85B5" d="M18.326 17.112h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H27.09c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V20.85h-1.666v-.938z"/><path id="name" fill="#181717" d="M51.562 22.362a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm8.61-1.904a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zM62.23 26.1c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm8.498.812V22.25c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm12.138-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H77.49c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="=" fill="#DBAF88" d="M93.212 20.514h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496V22.81z"/><path id=""Pete"" fill="#478964" d="M114.086 17.112h1.232l-.434 4.242h-.798v-4.242zm-2.898 0h1.232l-.434 4.242h-.798v-4.242zm7.672.098a7.07 7.07 0 011.148-.168c.41-.028.817-.042 1.218-.042.43 0 .866.042 1.309.126a3.55 3.55 0 011.211.462 2.6 2.6 0 01.889.924c.229.392.343.891.343 1.498 0 .597-.107 1.101-.322 1.512-.215.41-.5.744-.854 1.001-.355.257-.76.441-1.218.553a5.923 5.923 0 01-1.414.168h-.231a8.417 8.417 0 01-.679-.028 2.406 2.406 0 01-.238-.028v3.724h-1.162V17.21zm2.394.798c-.243 0-.476.005-.7.014-.224.01-.401.033-.532.07v4.06c.047.019.121.03.224.035l.322.014c.112.005.22.007.322.007h.224c.317 0 .632-.03.945-.091.313-.06.595-.17.847-.329s.455-.383.609-.672c.154-.29.231-.658.231-1.106 0-.383-.072-.702-.217-.959a1.735 1.735 0 00-.574-.616 2.419 2.419 0 00-.805-.329 4.182 4.182 0 00-.896-.098zm12.012 8.022c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V20.85h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm8.974-3.556h1.232l-.434 4.242h-.798v-4.242zm-2.898 0h1.232l-.434 4.242h-.798v-4.242z"/><path id=";" fill="#7E7C7B" d="M162.666 26.142c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="returnfunction" fill="#1C85B5" d="M17.584 60.912v-.938h1.89V54.85h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H27.09c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V54.85h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H48.09l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V54.85h-.896v-.938zm4.242 7v-.938h1.89V54.85h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm16.73 4.55v-.938h1.666V54.85h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H79.45v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H90.09l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V54.85h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V54.85h-1.666v-.938zm9.436 7v-.938h2.436V54.85h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M148.708 63.88a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert(name);" fill="#C06334" d="M34.972 71.458a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zM37.03 77.1c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm6.496-8.988h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H52.29c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V71.85h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V71.85h-1.666v-.938zm14.014 9.968a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848S77 75.509 77 74.51c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm3.654-7.518a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm8.61-1.904a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zM95.83 77.1c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm8.498.812V73.25c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm12.138-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.95 8.246a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="};}" fill="#7E7C7B" d="M21.98 95.836c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM5.18 112.836c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H.98v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H.98v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="let" fill="#1C85B5" d="M1.526 136.112H3.85v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H1.526v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H10.29c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="name" fill="#181717" d="M34.762 141.362a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm8.61-1.904a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm8.498.812v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm12.138-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H60.69c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="=" fill="#DBAF88" d="M76.412 139.514h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id=""John"" fill="#478964" d="M97.286 136.112h1.232l-.434 4.242h-.798v-4.242zm-2.898 0h1.232l-.434 4.242h-.798v-4.242zm7.63 0h5.544v6.328c0 1.176-.28 2.072-.84 2.688-.56.616-1.367.924-2.422.924a4.603 4.603 0 01-1.526-.259 3.593 3.593 0 01-.581-.259 1.437 1.437 0 01-.357-.266l.504-.924c.103.065.217.135.343.21.126.075.266.147.42.217a2.756 2.756 0 001.169.245c.7 0 1.23-.22 1.589-.658.36-.439.539-1.148.539-2.128v-5.082h-4.382v-1.036zm7.84 6.3c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm6.496-6.3h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547v4.032h-1.092v-3.822c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721v4.452h-1.078v-8.862h-1.106v-.938zm9.646 5.25a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm12.124-5.25h1.232l-.434 4.242h-.798v-4.242zm-2.898 0h1.232l-.434 4.242h-.798v-4.242z"/><path id=";" fill="#7E7C7B" d="M145.866 145.142c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="let" fill="#1C85B5" d="M1.526 153.112H3.85v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H1.526v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H10.29c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="work" fill="#181717" d="M38.206 157.452l1.4 3.892h.098l.994-5.432h1.022l-1.372 7h-1.246l-1.344-3.864h-.042l-1.4 3.864H35.07l-1.47-7h1.078l1.134 5.46h.084l1.358-3.92h.952zm4.452 1.96c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.364 3.5v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.812-9.8h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H65.03l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938z"/><path id="=" fill="#DBAF88" d="M76.412 156.514h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="makeWorker" fill="#181717" d="M95.928 162.912v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.166-8.988h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.37 3.43l.07.91h.098l.196-.994 1.036-3.822h.924l.966 3.794.252 1.036h.07l.084-.952 1.036-6.958h1.036l-1.512 9.8h-1.162l-1.134-3.976-.14-.952h-.042l-.154.966-1.134 3.962h-1.176l-1.526-9.8h1.106l1.106 6.986zm6.776-.686c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.364 3.5v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.812-9.8h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="();" fill="#7E7C7B" d="M182.308 165.88a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="work();" fill="#C06334" d="M4.606 191.452l1.4 3.892h.098l.994-5.432H8.12l-1.372 7H5.502l-1.344-3.864h-.042l-1.4 3.864H1.47l-1.47-7h1.078l1.134 5.46h.084l1.358-3.92h.952zm4.452 1.96c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.364 3.5v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.812-9.8h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H31.43l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm14.112 12.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848S35 194.509 35 193.51c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="//Pete" fill="#767D89" d="M73.374 186.944l.882.392-5.11 11.536-.882-.392 5.11-11.536zm8.4 0l.882.392-5.11 11.536-.882-.392 5.11-11.536zm11.886.266a7.07 7.07 0 011.148-.168c.41-.028.817-.042 1.218-.042.43 0 .866.042 1.309.126a3.55 3.55 0 011.211.462 2.6 2.6 0 01.889.924c.229.392.343.891.343 1.498 0 .597-.107 1.101-.322 1.512-.215.41-.5.744-.854 1.001-.355.257-.76.441-1.218.553a5.923 5.923 0 01-1.414.168h-.231a8.417 8.417 0 01-.679-.028 2.406 2.406 0 01-.238-.028v3.724H93.66v-9.702zm2.394.798c-.243 0-.476.005-.7.014-.224.01-.401.033-.532.07v4.06c.047.019.121.03.224.035l.322.014c.112.005.22.007.322.007h.224c.317 0 .632-.03.945-.091.313-.06.595-.17.847-.329s.455-.383.609-.672c.154-.29.231-.658.231-1.106 0-.383-.072-.702-.217-.959a1.735 1.735 0 00-.574-.616 2.419 2.419 0 00-.805-.329 4.182 4.182 0 00-.896-.098zm12.012 8.022c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/></g><path id="Rectangle-1-Copy-3" fill="#C06334" d="M0 95V82l9 6.5"/><g id="Group-3" transform="translate(494 64)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h183v47H1z"/><text id="makeWorker:-function" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="10" y="19">makeWorker: function</tspan> <tspan x="10" y="34">name: "John"</tspan></text></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(302 30)"><path id="Line" d="M.274 1h16.452"/><path id="Line-Copy" d="M.274 101h16.452"/><path id="Line-Copy-2" d="M18.5 1.133v99.734"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(482 10)"><path id="Line-Copy-2" d="M2.333 2.333v205.334"/></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="translate(186 69)"><path id="Line" d="M.355.422h8.394"/><path id="Line-Copy" d="M.355 37.578h8.394"/><path id="Line-Copy-2" d="M9.167.422v37.156"/></g><g id="Group-2-Copy" transform="translate(201 72)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h68v27H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="10" y="18"><empty></tspan></text></g><g id="Group" transform="translate(684 66)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-Copy" transform="translate(440 66)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-Copy-2" transform="translate(279 65)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M16.5 15.5l14 7-14 7v-6h-14v-2h14v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="723" y="91">null</tspan></text><g id="Group-3" transform="translate(326 71)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h109v28H1z"/><text id="name:-"Pete"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="5" y="18.51">name: "Pete"</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/2-closure-variable-access/solution.md b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/solution.md new file mode 100644 index 000000000..0a522132f --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/solution.md @@ -0,0 +1,9 @@ +The answer is: **Pete**. + +The `work()` function in the code below gets `name` from the place of its origin through the outer lexical environment reference: + + + +So, the result is `"Pete"` here. + +But if there were no `let name` in `makeWorker()`, then the search would go outside and take the global variable as we can see from the chain above. In that case the result would be `"John"`. diff --git a/1-js/06-advanced-functions/03-closure/2-closure-variable-access/task.md b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/task.md new file mode 100644 index 000000000..d12a385c8 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/2-closure-variable-access/task.md @@ -0,0 +1,29 @@ +importance: 5 + +--- + +# Which variables are available? + +The function `makeWorker` below makes another function and returns it. That new function can be called from somewhere else. + +Will it have access to the outer variables from its creation place, or the invocation place, or both? + +```js +function makeWorker() { + let name = "Pete"; + + return function() { + alert(name); + }; +} + +let name = "John"; + +// create a function +let work = makeWorker(); + +// call it +work(); // what will it show? +``` + +Which value it will show? "Pete" or "John"? diff --git a/1-js/06-advanced-functions/03-closure/1-counter-independent/solution.md b/1-js/06-advanced-functions/03-closure/3-counter-independent/solution.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/1-counter-independent/solution.md rename to 1-js/06-advanced-functions/03-closure/3-counter-independent/solution.md diff --git a/1-js/06-advanced-functions/03-closure/1-counter-independent/task.md b/1-js/06-advanced-functions/03-closure/3-counter-independent/task.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/1-counter-independent/task.md rename to 1-js/06-advanced-functions/03-closure/3-counter-independent/task.md diff --git a/1-js/06-advanced-functions/03-closure/2-counter-object-independent/solution.md b/1-js/06-advanced-functions/03-closure/4-counter-object-independent/solution.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/2-counter-object-independent/solution.md rename to 1-js/06-advanced-functions/03-closure/4-counter-object-independent/solution.md diff --git a/1-js/06-advanced-functions/03-closure/2-counter-object-independent/task.md b/1-js/06-advanced-functions/03-closure/4-counter-object-independent/task.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/2-counter-object-independent/task.md rename to 1-js/06-advanced-functions/03-closure/4-counter-object-independent/task.md diff --git a/1-js/06-advanced-functions/03-closure/3-function-in-if/solution.md b/1-js/06-advanced-functions/03-closure/5-function-in-if/solution.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/3-function-in-if/solution.md rename to 1-js/06-advanced-functions/03-closure/5-function-in-if/solution.md diff --git a/1-js/06-advanced-functions/03-closure/3-function-in-if/task.md b/1-js/06-advanced-functions/03-closure/5-function-in-if/task.md similarity index 65% rename from 1-js/06-advanced-functions/03-closure/3-function-in-if/task.md rename to 1-js/06-advanced-functions/03-closure/5-function-in-if/task.md index d0dbbeb11..4e386eec5 100644 --- a/1-js/06-advanced-functions/03-closure/3-function-in-if/task.md +++ b/1-js/06-advanced-functions/03-closure/5-function-in-if/task.md @@ -1,7 +1,9 @@ +importance: 5 +--- # Function in if -Look at the code. What will be result of the call at the last line? +Look at the code. What will be the result of the call at the last line? ```js run let phrase = "Hello"; diff --git a/1-js/06-advanced-functions/03-closure/4-closure-sum/solution.md b/1-js/06-advanced-functions/03-closure/6-closure-sum/solution.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/4-closure-sum/solution.md rename to 1-js/06-advanced-functions/03-closure/6-closure-sum/solution.md diff --git a/1-js/06-advanced-functions/03-closure/4-closure-sum/task.md b/1-js/06-advanced-functions/03-closure/6-closure-sum/task.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/4-closure-sum/task.md rename to 1-js/06-advanced-functions/03-closure/6-closure-sum/task.md diff --git a/1-js/06-advanced-functions/03-closure/7-let-scope/solution.md b/1-js/06-advanced-functions/03-closure/7-let-scope/solution.md new file mode 100644 index 000000000..b16b35290 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/7-let-scope/solution.md @@ -0,0 +1,40 @@ +The result is: **error**. + +Try running it: + +```js run +let x = 1; + +function func() { +*!* + console.log(x); // ReferenceError: Cannot access 'x' before initialization +*/!* + let x = 2; +} + +func(); +``` + +In this example we can observe the peculiar difference between a "non-existing" and "uninitialized" variable. + +As you may have read in the article [](info:closure), a variable starts in the "uninitialized" state from the moment when the execution enters a code block (or a function). And it stays uninitalized until the corresponding `let` statement. + +In other words, a variable technically exists, but can't be used before `let`. + +The code above demonstrates it. + +```js +function func() { +*!* + // the local variable x is known to the engine from the beginning of the function, + // but "uninitialized" (unusable) until let ("dead zone") + // hence the error +*/!* + + console.log(x); // ReferenceError: Cannot access 'x' before initialization + + let x = 2; +} +``` + +This zone of temporary unusability of a variable (from the beginning of the code block till `let`) is sometimes called the "dead zone". diff --git a/1-js/06-advanced-functions/03-closure/7-let-scope/task.md b/1-js/06-advanced-functions/03-closure/7-let-scope/task.md new file mode 100644 index 000000000..fb7445e66 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/7-let-scope/task.md @@ -0,0 +1,21 @@ +importance: 4 + +--- + +# Is variable visible? + +What will be the result of this code? + +```js +let x = 1; + +function func() { + console.log(x); // ? + + let x = 2; +} + +func(); +``` + +P.S. There's a pitfall in this task. The solution is not obvious. diff --git a/1-js/06-advanced-functions/03-closure/7-sort-by-field/solution.md b/1-js/06-advanced-functions/03-closure/7-sort-by-field/solution.md deleted file mode 100644 index bd57085ea..000000000 --- a/1-js/06-advanced-functions/03-closure/7-sort-by-field/solution.md +++ /dev/null @@ -1,22 +0,0 @@ - - -```js run -let users = [ - { name: "John", age: 20, surname: "Johnson" }, - { name: "Pete", age: 18, surname: "Peterson" }, - { name: "Ann", age: 19, surname: "Hathaway" } -]; - -*!* -function byField(field) { - return (a, b) => a[field] > b[field] ? 1 : -1; -} -*/!* - -users.sort(byField('name')); -users.forEach(user => alert(user.name)); // Ann, John, Pete - -users.sort(byField('age')); -users.forEach(user => alert(user.name)); // Pete, Ann, John -``` - diff --git a/1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/solution.js b/1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/solution.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/solution.js rename to 1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/solution.js diff --git a/1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/source.js b/1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/source.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/source.js rename to 1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/source.js diff --git a/1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/test.js b/1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/test.js similarity index 100% rename from 1-js/06-advanced-functions/03-closure/6-filter-through-function/_js.view/test.js rename to 1-js/06-advanced-functions/03-closure/8-filter-through-function/_js.view/test.js diff --git a/1-js/06-advanced-functions/03-closure/6-filter-through-function/solution.md b/1-js/06-advanced-functions/03-closure/8-filter-through-function/solution.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/6-filter-through-function/solution.md rename to 1-js/06-advanced-functions/03-closure/8-filter-through-function/solution.md diff --git a/1-js/06-advanced-functions/03-closure/6-filter-through-function/task.md b/1-js/06-advanced-functions/03-closure/8-filter-through-function/task.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/6-filter-through-function/task.md rename to 1-js/06-advanced-functions/03-closure/8-filter-through-function/task.md diff --git a/1-js/06-advanced-functions/03-closure/8-make-army/task.md b/1-js/06-advanced-functions/03-closure/8-make-army/task.md deleted file mode 100644 index ede8fd045..000000000 --- a/1-js/06-advanced-functions/03-closure/8-make-army/task.md +++ /dev/null @@ -1,35 +0,0 @@ -importance: 5 - ---- - -# Army of functions - -The following code creates an array of `shooters`. - -Every function is meant to output its number. But something is wrong... - -```js run -function makeArmy() { - let shooters = []; - - let i = 0; - while (i < 10) { - let shooter = function() { // shooter function - alert( i ); // should show its number - }; - shooters.push(shooter); - i++; - } - - return shooters; -} - -let army = makeArmy(); - -army[0](); // the shooter number 0 shows 10 -army[5](); // and number 5 also outputs 10... -// ... all shooters show 10 instead of their 0, 1, 2, 3... -``` - -Why all shooters show the same? Fix the code so that they work as intended. - diff --git a/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/solution.js b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/solution.js new file mode 100644 index 000000000..8a71c869d --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/solution.js @@ -0,0 +1,3 @@ +function byField(fieldName){ + return (a, b) => a[fieldName] > b[fieldName] ? 1 : -1; +} diff --git a/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/source.js b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/source.js new file mode 100644 index 000000000..23b433834 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/source.js @@ -0,0 +1,5 @@ +function byField(fieldName){ + + // Your code goes here. + +} diff --git a/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/test.js b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/test.js new file mode 100644 index 000000000..802f28c4d --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/9-sort-by-field/_js.view/test.js @@ -0,0 +1,39 @@ +describe("byField", function(){ + + let users = [ + { name: "John", age: 20, surname: "Johnson" }, + { name: "Pete", age: 18, surname: "Peterson" }, + { name: "Ann", age: 19, surname: "Hathaway" }, + ]; + + it("sorts users by name", function(){ + let nameSortedKey = [ + { name: "Ann", age: 19, surname: "Hathaway" }, + { name: "John", age: 20, surname: "Johnson"}, + { name: "Pete", age: 18, surname: "Peterson" }, + ]; + let nameSortedAnswer = users.sort(byField("name")); + assert.deepEqual(nameSortedKey, nameSortedAnswer); + }); + + it("sorts users by age", function(){ + let ageSortedKey = [ + { name: "Pete", age: 18, surname: "Peterson" }, + { name: "Ann", age: 19, surname: "Hathaway" }, + { name: "John", age: 20, surname: "Johnson"}, + ]; + let ageSortedAnswer = users.sort(byField("age")); + assert.deepEqual(ageSortedKey, ageSortedAnswer); + }); + + it("sorts users by surname", function(){ + let surnameSortedKey = [ + { name: "Ann", age: 19, surname: "Hathaway" }, + { name: "John", age: 20, surname: "Johnson"}, + { name: "Pete", age: 18, surname: "Peterson" }, + ]; + let surnameSortedAnswer = users.sort(byField("surname")); + assert.deepEqual(surnameSortedAnswer, surnameSortedKey); + }); + +}); diff --git a/1-js/06-advanced-functions/03-closure/9-sort-by-field/solution.md b/1-js/06-advanced-functions/03-closure/9-sort-by-field/solution.md new file mode 100644 index 000000000..8b1378917 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/9-sort-by-field/solution.md @@ -0,0 +1 @@ + diff --git a/1-js/06-advanced-functions/03-closure/7-sort-by-field/task.md b/1-js/06-advanced-functions/03-closure/9-sort-by-field/task.md similarity index 100% rename from 1-js/06-advanced-functions/03-closure/7-sort-by-field/task.md rename to 1-js/06-advanced-functions/03-closure/9-sort-by-field/task.md diff --git a/1-js/06-advanced-functions/03-closure/article.md b/1-js/06-advanced-functions/03-closure/article.md index e771b1cc4..ff6ca062d 100644 --- a/1-js/06-advanced-functions/03-closure/article.md +++ b/1-js/06-advanced-functions/03-closure/article.md @@ -1,25 +1,29 @@ -# Closure +# Variable scope, closure -JavaScript is a very function-oriented language. It gives us a lot of freedom. A function can be created at one moment, then copied to another variable or passed as an argument to another function and called from a totally different place later. +JavaScript is a very function-oriented language. It gives us a lot of freedom. A function can be created at any moment, passed as an argument to another function, and then called from a totally different place of code later. -We know that a function can access variables outside of it; this feature is used quite often. +We already know that a function can access variables outside of it ("outer" variables). -But what happens when an outer variable changes? Does a function get the most recent value or the one that existed when the function was created? +But what happens if outer variables change since a function is created? Will the function get newer values or the old ones? -Also, what happens when a function travels to another place in the code and is called from there -- does it get access to the outer variables of the new place? +And what if a function is passed along as an argument and called from another place of code, will it get access to outer variables at the new place? -Different languages behave differently here, and in this chapter we cover the behaviour of JavaScript. +Let's expand our knowledge to understand these scenarios and more complex ones. -## A couple of questions +```smart header="We'll talk about `let/const` variables here" +In JavaScript, there are 3 ways to declare a variable: `let`, `const` (the modern ones), and `var` (the remnant of the past). -Let's consider two situations to begin with, and then study the internal mechanics piece-by-piece, so that you'll be able to answer the following questions and more complex ones in the future. +- In this article we'll use `let` variables in examples. +- Variables, declared with `const`, behave the same, so this article is about `const` too. +- The old `var` has some notable differences, they will be covered in the article <info:var>. +``` -1. The function `sayHi` uses an external variable `name`. When the function runs, which value is it going to use? +## Code blocks - ```js - let name = "John"; +If a variable is declared inside a code block `{...}`, it's only visible inside that block. +<<<<<<< HEAD function sayHi() { alert("Hi, " + name); } @@ -165,39 +169,82 @@ Now we can give the answer to the first question from the beginning of the chapt That's because of the described mechanism. Old variable values are not saved anywhere. When a function wants them, it takes the current values from its own or an outer Lexical Environment. So the answer to the first question is `Pete`: +======= +For example: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js run -let name = "John"; +{ + // do some job with local variables that should not be seen outside + + let message = "Hello"; // only visible in this block -function sayHi() { - alert("Hi, " + name); + alert(message); // Hello } -name = "Pete"; // (*) +alert(message); // Error: message is not defined +``` + +We can use this to isolate a piece of code that does its own task, with variables that only belong to it: + +```js run +{ + // show message + let message = "Hello"; + alert(message); +} +{ + // show another message + let message = "Goodbye"; + alert(message); +} +``` + +````smart header="There'd be an error without blocks" +Please note, without separate blocks there would be an error, if we use `let` with the existing variable name: + +```js run +// show message +let message = "Hello"; +alert(message); + +// show another message *!* -sayHi(); // Pete +let message = "Goodbye"; // Error: variable already declared */!* +alert(message); ``` +```` +For `if`, `for`, `while` and so on, variables declared in `{...}` are also only visible inside: -The execution flow of the code above: +```js run +if (true) { + let phrase = "Hello!"; -1. The global Lexical Environment has `name: "John"`. -2. At the line `(*)` the global variable is changed, now it has `name: "Pete"`. -3. When the function `sayHi()`, is executed and takes `name` from outside. Here that's from the global Lexical Environment where it's already `"Pete"`. + alert(phrase); // Hello! +} +alert(phrase); // Error, no such variable! +``` -```smart header="One call -- one Lexical Environment" -Please note that a new function Lexical Environment is created each time a function runs. +Here, after `if` finishes, the `alert` below won't see the `phrase`, hence the error. -And if a function is called multiple times, then each invocation will have its own Lexical Environment, with local variables and parameters specific for that very run. -``` +That's great, as it allows us to create block-local variables, specific to an `if` branch. -```smart header="Lexical Environment is a specification object" -"Lexical Environment" is a specification object. We can't get this object in our code and manipulate it directly. JavaScript engines also may optimize it, discard variables that are unused to save memory and perform other internal tricks, but the visible behavior should be as described. +The similar thing holds true for `for` and `while` loops: + +```js run +for (let i = 0; i < 3; i++) { + // the variable i is only visible inside this for + alert(i); // 0, then 1, then 2 +} + +alert(i); // Error, no such variable ``` +Visually, `let i` is outside of `{...}`. But the `for` construct is special here: the variable, declared inside it, is considered a part of the block. ## Nested functions @@ -223,32 +270,16 @@ function sayHiBye(firstName, lastName) { Here the *nested* function `getFullName()` is made for convenience. It can access the outer variables and so can return the full name. Nested functions are quite common in JavaScript. -What's much more interesting, a nested function can be returned: either as a property of a new object (if the outer function creates an object with methods) or as a result by itself. It can then be used somewhere else. No matter where, it still has access to the same outer variables. +What's much more interesting, a nested function can be returned: either as a property of a new object or as a result by itself. It can then be used somewhere else. No matter where, it still has access to the same outer variables. -For instance, here the nested function is assigned to the new object by the [constructor function](info:constructor-new): - -```js run -// constructor function returns a new object -function User(name) { - - // the object method is created as a nested function - this.sayHi = function() { - alert(name); - }; -} - -let user = new User("John"); -user.sayHi(); // the method "sayHi" code has access to the outer "name" -``` - -And here we just create and return a "counting" function: +Below, `makeCounter` creates the "counter" function that returns the next number on each invocation: ```js run function makeCounter() { let count = 0; return function() { - return count++; // has access to the outer counter + return count++; }; } @@ -259,80 +290,171 @@ alert( counter() ); // 1 alert( counter() ); // 2 ``` -Let's go on with the `makeCounter` example. It creates the "counter" function that returns the next number on each invocation. Despite being simple, slightly modified variants of that code have practical uses, for instance, as a [pseudorandom number generator](https://en.wikipedia.org/wiki/Pseudorandom_number_generator), and more. +Despite being simple, slightly modified variants of that code have practical uses, for instance, as a [random number generator](https://en.wikipedia.org/wiki/Pseudorandom_number_generator) to generate random values for automated tests. -How does the counter work internally? +How does this work? If we create multiple counters, will they be independent? What's going on with the variables here? -When the inner function runs, the variable in `count++` is searched from inside out. For the example above, the order will be: +Understanding such things is great for the overall knowledge of JavaScript and beneficial for more complex scenarios. So let's go a bit in-depth. +<<<<<<< HEAD  +======= +## Lexical Environment +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -1. The locals of the nested function... -2. The variables of the outer function... -3. And so on until it reaches global variables. +```warn header="Here be dragons!" +The in-depth technical explanation lies ahead. -In this example `count` is found on step `2`. When an outer variable is modified, it's changed where it's found. So `count++` finds the outer variable and increases it in the Lexical Environment where it belongs. Like if we had `let count = 1`. +As far as I'd like to avoid low-level language details, any understanding without them would be lacking and incomplete, so get ready. +``` -Here are two questions to consider: +For clarity, the explanation is split into multiple steps. -1. Can we somehow reset the counter `count` from the code that doesn't belong to `makeCounter`? E.g. after `alert` calls in the example above. -2. If we call `makeCounter()` multiple times -- it returns many `counter` functions. Are they independent or do they share the same `count`? +### Step 1. Variables -Try to answer them before you continue reading. +In JavaScript, every running function, code block `{...}`, and the script as a whole have an internal (hidden) associated object known as the *Lexical Environment*. -... +The Lexical Environment object consists of two parts: -All done? +1. *Environment Record* -- an object that stores all local variables as its properties (and some other information like the value of `this`). +2. A reference to the *outer lexical environment*, the one associated with the outer code. -Okay, let's go over the answers. +**A "variable" is just a property of the special internal object, `Environment Record`. "To get or change a variable" means "to get or change a property of that object".** -1. There is no way: `count` is a local function variable, we can't access it from the outside. -2. For every call to `makeCounter()` a new function Lexical Environment is created, with its own `count`. So the resulting `counter` functions are independent. +In this simple code without functions, there is only one Lexical Environment: -Here's the demo: + -```js run +This is the so-called *global* Lexical Environment, associated with the whole script. + +On the picture above, the rectangle means Environment Record (variable store) and the arrow means the outer reference. The global Lexical Environment has no outer reference, that's why the arrow points to `null`. + +As the code starts executing and goes on, the Lexical Environment changes. + +Here's a little bit longer code: + + + +Rectangles on the right-hand side demonstrate how the global Lexical Environment changes during the execution: + +1. When the script starts, the Lexical Environment is pre-populated with all declared variables. + - Initially, they are in the "Uninitialized" state. That's a special internal state, it means that the engine knows about the variable, but it cannot be referenced until it has been declared with `let`. It's almost the same as if the variable didn't exist. +2. Then `let phrase` definition appears. There's no assignment yet, so its value is `undefined`. We can use the variable from this point forward. +3. `phrase` is assigned a value. +4. `phrase` changes the value. + +Everything looks simple for now, right? + +- A variable is a property of a special internal object, associated with the currently executing block/function/script. +- Working with variables is actually working with the properties of that object. + +```smart header="Lexical Environment is a specification object" +"Lexical Environment" is a specification object: it only exists "theoretically" in the [language specification](https://tc39.es/ecma262/#sec-lexical-environments) to describe how things work. We can't get this object in our code and manipulate it directly. + +JavaScript engines also may optimize it, discard variables that are unused to save memory and perform other internal tricks, as long as the visible behavior remains as described. +``` + +### Step 2. Function Declarations + +A function is also a value, like a variable. + +**The difference is that a Function Declaration is instantly fully initialized.** + +When a Lexical Environment is created, a Function Declaration immediately becomes a ready-to-use function (unlike `let`, that is unusable till the declaration). + +That's why we can use a function, declared as Function Declaration, even before the declaration itself. + +For example, here's the initial state of the global Lexical Environment when we add a function: + + + +Naturally, this behavior only applies to Function Declarations, not Function Expressions where we assign a function to a variable, such as `let say = function(name)...`. + +### Step 3. Inner and outer Lexical Environment + +When a function runs, at the beginning of the call, a new Lexical Environment is created automatically to store local variables and parameters of the call. + +For instance, for `say("John")`, it looks like this (the execution is at the line, labelled with an arrow): + +<!-- + ```js + let phrase = "Hello"; + + function say(name) { + alert( `${phrase}, ${name}` ); + } + + say("John"); // Hello, John + ```--> + + + +During the function call we have two Lexical Environments: the inner one (for the function call) and the outer one (global): + +- The inner Lexical Environment corresponds to the current execution of `say`. It has a single property: `name`, the function argument. We called `say("John")`, so the value of the `name` is `"John"`. +- The outer Lexical Environment is the global Lexical Environment. It has the `phrase` variable and the function itself. + +The inner Lexical Environment has a reference to the `outer` one. + +**When the code wants to access a variable -- the inner Lexical Environment is searched first, then the outer one, then the more outer one and so on until the global one.** + +If a variable is not found anywhere, that's an error in strict mode (without `use strict`, an assignment to a non-existing variable creates a new global variable, for compatibility with old code). + +In this example the search proceeds as follows: + +- For the `name` variable, the `alert` inside `say` finds it immediately in the inner Lexical Environment. +- When it wants to access `phrase`, then there is no `phrase` locally, so it follows the reference to the outer Lexical Environment and finds it there. + + + + +### Step 4. Returning a function + +Let's return to the `makeCounter` example. + +```js function makeCounter() { let count = 0; + return function() { return count++; }; } -let counter1 = makeCounter(); -let counter2 = makeCounter(); - -alert( counter1() ); // 0 -alert( counter1() ); // 1 - -alert( counter2() ); // 0 (independent) +let counter = makeCounter(); ``` +At the beginning of each `makeCounter()` call, a new Lexical Environment object is created, to store variables for this `makeCounter` run. -Hopefully, the situation with outer variables is quite clear for you now. But in more complex situations a deeper understanding of internals may be required. So let's dive deeper. +So we have two nested Lexical Environments, just like in the example above: -## Environments in detail + -Now that you understand how closures work generally, that's already very good. +What's different is that, during the execution of `makeCounter()`, a tiny nested function is created of only one line: `return count++`. We don't run it yet, only create. -Here's what's going on in the `makeCounter` example step-by-step, follow it to make sure that you know things in the very detail. +All functions remember the Lexical Environment in which they were made. Technically, there's no magic here: all functions have the hidden property named `[[Environment]]`, that keeps the reference to the Lexical Environment where the function was created: -Please note the additional `[[Environment]]` property is covered here. We didn't mention it before for simplicity. + -1. When the script has just started, there is only global Lexical Environment: +So, `counter.[[Environment]]` has the reference to `{count: 0}` Lexical Environment. That's how the function remembers where it was created, no matter where it's called. The `[[Environment]]` reference is set once and forever at function creation time. +<<<<<<< HEAD  +======= +Later, when `counter()` is called, a new Lexical Environment is created for the call, and its outer Lexical Environment reference is taken from `counter.[[Environment]]`: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b - At that starting moment there is only `makeCounter` function, because it's a Function Declaration. It did not run yet. + - **All functions "on birth" receive a hidden property `[[Environment]]` with a reference to the Lexical Environment of their creation.** We didn't talk about it yet, but that's how the function knows where it was made. +Now when the code inside `counter()` looks for `count` variable, it first searches its own Lexical Environment (empty, as there are no local variables there), then the Lexical Environment of the outer `makeCounter()` call, where it finds and changes it. - Here, `makeCounter` is created in the global Lexical Environment, so `[[Environment]]` keeps a reference to it. +**A variable is updated in the Lexical Environment where it lives.** - In other words, a function is "imprinted" with a reference to the Lexical Environment where it was born. And `[[Environment]]` is the hidden function property that has that reference. +Here's the state after the execution: -2. The code runs on, the new global variable `counter` is declared and for its value `makeCounter()` is called. Here's a snapshot of the moment when the execution is on the first line inside `makeCounter()`: + +<<<<<<< HEAD  At the moment of the call of `makeCounter()`, the Lexical Environment is created, to hold its variables and arguments. @@ -390,15 +512,21 @@ So, the result is `"Pete"` here. But if there were no `let name` in `makeWorker()`, then the search would go outside and take the global variable as we can see from the chain above. In that case it would be `"John"`. ```smart header="Closures" +======= +If we call `counter()` multiple times, the `count` variable will be increased to `2`, `3` and so on, at the same place. + +```smart header="Closure" +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b There is a general programming term "closure", that developers generally should know. -A [closure](https://en.wikipedia.org/wiki/Closure_(computer_programming)) is a function that remembers its outer variables and can access them. In some languages, that's not possible, or a function should be written in a special way to make it happen. But as explained above, in JavaScript, all functions are naturally closures (there is only one exclusion, to be covered in <info:new-function>). +A [closure](https://en.wikipedia.org/wiki/Closure_(computer_programming)) is a function that remembers its outer variables and can access them. In some languages, that's not possible, or a function should be written in a special way to make it happen. But as explained above, in JavaScript, all functions are naturally closures (there is only one exception, to be covered in <info:new-function>). -That is: they automatically remember where they were created using a hidden `[[Environment]]` property, and all of them can access outer variables. +That is: they automatically remember where they were created using a hidden `[[Environment]]` property, and then their code can access outer variables. -When on an interview, a frontend developer gets a question about "what's a closure?", a valid answer would be a definition of the closure and an explanation that all functions in JavaScript are closures, and maybe few more words about technical details: the `[[Environment]]` property and how Lexical Environments work. +When on an interview, a frontend developer gets a question about "what's a closure?", a valid answer would be a definition of the closure and an explanation that all functions in JavaScript are closures, and maybe a few more words about technical details: the `[[Environment]]` property and how Lexical Environments work. ``` +<<<<<<< HEAD ## Code blocks and loops, IIFE The examples above concentrated on functions. But a Lexical Environment exists for any code block `{...}`. @@ -541,38 +669,32 @@ There exist other ways besides parentheses to tell JavaScript that we mean a Fun In all the above cases we declare a Function Expression and run it immediately. +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Garbage collection -Usually, a Lexical Environment is cleaned up and deleted after the function run. For instance: - -```js -function f() { - let value1 = 123; - let value2 = 456; -} +Usually, a Lexical Environment is removed from memory with all the variables after the function call finishes. That's because there are no references to it. As any JavaScript object, it's only kept in memory while it's reachable. -f(); -``` +However, if there's a nested function that is still reachable after the end of a function, then it has `[[Environment]]` property that references the lexical environment. -Here two values are technically the properties of the Lexical Environment. But after `f()` finishes that Lexical Environment becomes unreachable, so it's deleted from the memory. +In that case the Lexical Environment is still reachable even after the completion of the function, so it stays alive. -...But if there's a nested function that is still reachable after the end of `f`, then its `[[Environment]]` reference keeps the outer lexical environment alive as well: +For example: ```js function f() { let value = 123; - function g() { alert(value); } - -*!* - return g; -*/!* + return function() { + alert(value); + } } -let g = f(); // g is reachable, and keeps the outer lexical environment in memory +let g = f(); // g.[[Environment]] stores a reference to the Lexical Environment +// of the corresponding f() call ``` -Please note that if `f()` is called many times, and resulting functions are saved, then the corresponding Lexical Environment objects will also be retained in memory. All 3 of them in the code below: +Please note that if `f()` is called many times, and resulting functions are saved, then all corresponding Lexical Environment objects will also be retained in memory. In the code below, all 3 of them: ```js function f() { @@ -583,25 +705,23 @@ function f() { // 3 functions in array, every one of them links to Lexical Environment // from the corresponding f() run -// LE LE LE let arr = [f(), f(), f()]; ``` A Lexical Environment object dies when it becomes unreachable (just like any other object). In other words, it exists only while there's at least one nested function referencing it. -In the code below, after `g` becomes unreachable, enclosing Lexical Environment (and hence the `value`) is cleaned from memory; +In the code below, after the nested function is removed, its enclosing Lexical Environment (and hence the `value`) is cleaned from memory: ```js function f() { let value = 123; - function g() { alert(value); } - - return g; + return function() { + alert(value); + } } -let g = f(); // while g is alive -// there corresponding Lexical Environment lives +let g = f(); // while g function exists, the value stays in memory g = null; // ...and now the memory is cleaned up ``` @@ -610,9 +730,9 @@ g = null; // ...and now the memory is cleaned up As we've seen, in theory while a function is alive, all outer variables are also retained. -But in practice, JavaScript engines try to optimize that. They analyze variable usage and if it's easy to see that an outer variable is not used -- it is removed. +But in practice, JavaScript engines try to optimize that. They analyze variable usage and if it's obvious from the code that an outer variable is not used -- it is removed. -**An important side effect in V8 (Chrome, Opera) is that such variable will become unavailable in debugging.** +**An important side effect in V8 (Chrome, Edge, Opera) is that such variable will become unavailable in debugging.** Try running the example below in Chrome with the Developer Tools open. @@ -623,7 +743,7 @@ function f() { let value = Math.random(); function g() { - debugger; // in console: type alert( value ); No such variable! + debugger; // in console: type alert(value); No such variable! } return g; @@ -644,7 +764,7 @@ function f() { let value = "the closest value"; function g() { - debugger; // in console: type alert( value ); Surprise! + debugger; // in console: type alert(value); Surprise! } return g; @@ -654,9 +774,6 @@ let g = f(); g(); ``` -```warn header="See ya!" -This feature of V8 is good to know. If you are debugging with Chrome/Opera, sooner or later you will meet it. +This feature of V8 is good to know. If you are debugging with Chrome/Edge/Opera, sooner or later you will meet it. -That is not a bug in the debugger, but rather a special feature of V8. Perhaps it will be changed sometime. -You always can check for it by running the examples on this page. -``` +That is not a bug in the debugger, but rather a special feature of V8. Perhaps it will be changed sometime. You can always check for it by running the examples on this page. diff --git a/1-js/06-advanced-functions/03-closure/closure-function-declaration.svg b/1-js/06-advanced-functions/03-closure/closure-function-declaration.svg new file mode 100644 index 000000000..3ef787875 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-function-declaration.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="678" height="169" viewBox="0 0 678 169"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-function-declaration.svg"><g id="Group" transform="translate(533 2)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M30 14l14 7-14 7v-6H-1v-2h31v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="587" y="26">null</tspan></text><g id="let-phrase-=-"Hello"" fill-rule="nonzero" transform="translate(19 71)"><path id="let" fill="#1C85B5" d="M.546 0H2.87v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197V.938H.546V0zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155A4.767 4.767 0 018.162 6.3c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H9.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666V1.414l1.092-.308V2.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.738h-1.666V2.8z"/><path id="phrase" fill="#181717" d="M33.824 5.25a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938V2.8h1.862l.14.98h.07c.252-.327.572-.6.959-.819.387-.22.861-.329 1.421-.329.999 0 1.75.275 2.254.826.504.55.756 1.451.756 2.702 0 .588-.086 1.118-.259 1.589a3.34 3.34 0 01-.735 1.197 3.24 3.24 0 01-1.148.756 3.999 3.999 0 01-1.484.266c-.196 0-.371-.007-.525-.021a4.613 4.613 0 01-.42-.056 3.31 3.31 0 01-.371-.091 5.103 5.103 0 01-.392-.14v2.94h-1.078V5.25zm3.22-1.596c-.28 0-.544.051-.791.154a2.36 2.36 0 00-1.127.966c-.121.205-.196.41-.224.616v3.192c.205.14.436.25.693.329.257.08.595.119 1.015.119.747 0 1.344-.245 1.792-.735.448-.49.672-1.206.672-2.149 0-.793-.156-1.407-.469-1.841-.313-.434-.833-.651-1.561-.651zM40.936 0h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V9.8h-1.092V5.978c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V9.8h-1.078V.938h-1.106V0zm9.268 9.8v-.938h1.89V3.738h-1.89V2.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688V9.8h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM72.212 7.91a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm9.674 1.008c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H76.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="=" fill="#DBAF88" d="M92.232 3.402h6.496V4.41h-6.496V3.402zm0 2.296h6.496v1.008h-6.496V5.698z"/><path id=""Hello"" fill="#478964" d="M113.106 0h1.232l-.434 4.242h-.798V0zm-2.898 0h1.232l-.434 4.242h-.798V0zm12.67 5.306h-4.396V9.8h-1.162V0h1.162v4.27h4.396V0h1.162v9.8h-1.162V5.306zm9.408 3.612c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM134.946 0h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197V.938h-1.232V0zm8.4 0h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197V.938h-1.232V0zm7.532 6.3c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zM163.506 0h1.232l-.434 4.242h-.798V0zm-2.898 0h1.232l-.434 4.242h-.798V0z"/><path id=";" fill="#7E7C7B" d="M170.086 9.03c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="function" fill="#1C85B5" d="M.112 43.8v-.938h1.666v-5.124H.112V36.8h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V36.8h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896V36.8zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V43.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V43.8h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308V36.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V36.8zm9.436 7v-.938h2.436v-5.124H42.35V36.8h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V43.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V43.8h-1.078v-4.55z"/><path id="say" fill="#181717" d="M80.612 41.91a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm3.78-4.564a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm9.184-.266h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922z"/><path id="(" fill="#7E7C7B" d="M105.728 46.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="name" fill="#181717" d="M109.382 39.25a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V43.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V43.8h-1.078v-4.55zm8.61-1.904a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm8.498.812v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98V43.8h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889V43.8h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12V43.8h-1.064zm12.138-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="){" fill="#7E7C7B" d="M143.682 45.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="alert" fill="#181717" d="M17.192 54.346a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM25.746 51h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V51zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89V53.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308V53.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666V53.8z"/><path id="(" fill="#7E7C7B" d="M63.728 63.768a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246A6.005 6.005 0 0163.728 51l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="`" fill="#478964" d="M79.66 52.806h-.77l-1.666-2.086h1.526z"/><path id="${" fill="#7E7C7B" d="M86.562 60.968a7.73 7.73 0 01-1.365-.133c-.397-.08-.716-.184-.959-.315l.378-1.022c.187.112.441.212.763.301s.716.142 1.183.161v-3.78c-.28-.14-.558-.29-.833-.448a3.391 3.391 0 01-.735-.567 2.573 2.573 0 01-.518-.777c-.13-.299-.196-.653-.196-1.064 0-.672.189-1.223.567-1.652.378-.43.95-.695 1.715-.798V49.6h.952v1.232c.476.019.866.063 1.169.133.303.07.576.161.819.273l-.336.98a3.584 3.584 0 00-.672-.245 4.736 4.736 0 00-.98-.133v3.458c.29.15.579.31.868.483.29.173.548.371.777.595.229.224.413.485.553.784.14.299.21.649.21 1.05 0 .728-.203 1.328-.609 1.799-.406.471-1.006.772-1.799.903V62.2h-.952v-1.232zm2.198-2.66c0-.224-.04-.425-.119-.602a1.724 1.724 0 00-.315-.476 2.327 2.327 0 00-.462-.378 6.612 6.612 0 00-.574-.322v3.402c.43-.075.782-.243 1.057-.504.275-.261.413-.635.413-1.12zm-3.318-5.054c0 .401.128.728.385.98s.576.48.959.686v-3.066c-.495.056-.842.22-1.043.49-.2.27-.301.574-.301.91zm8.918 7.098c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434A1.641 1.641 0 0092.4 58v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="phrase" fill="#181717" d="M101.024 56.25a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.862l.14.98h.07c.252-.327.572-.6.959-.819.387-.22.861-.329 1.421-.329.999 0 1.75.275 2.254.826.504.55.756 1.451.756 2.702 0 .588-.086 1.118-.259 1.589a3.34 3.34 0 01-.735 1.197 3.24 3.24 0 01-1.148.756 3.999 3.999 0 01-1.484.266c-.196 0-.371-.007-.525-.021a4.613 4.613 0 01-.42-.056 3.31 3.31 0 01-.371-.091 5.103 5.103 0 01-.392-.14v2.94h-1.078v-7.35zm3.22-1.596c-.28 0-.544.051-.791.154a2.36 2.36 0 00-1.127.966c-.121.205-.196.41-.224.616v3.192c.205.14.436.25.693.329.257.08.595.119 1.015.119.747 0 1.344-.245 1.792-.735.448-.49.672-1.206.672-2.149 0-.793-.156-1.407-.469-1.841-.313-.434-.833-.651-1.561-.651zM108.136 51h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V60.8h-1.092v-3.822c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V60.8h-1.078v-8.862h-1.106V51zm9.268 9.8v-.938h1.89v-5.124h-1.89V53.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm10.962-1.078a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm9.674 1.008c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="}" fill="#7E7C7B" d="M155.4 61.724c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V60.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V58c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="," fill="#478964" d="M161.686 60.03c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679z"/><path id="${" fill="#7E7C7B" d="M178.962 60.968a7.73 7.73 0 01-1.365-.133c-.397-.08-.716-.184-.959-.315l.378-1.022c.187.112.441.212.763.301s.716.142 1.183.161v-3.78c-.28-.14-.558-.29-.833-.448a3.391 3.391 0 01-.735-.567 2.573 2.573 0 01-.518-.777c-.13-.299-.196-.653-.196-1.064 0-.672.189-1.223.567-1.652.378-.43.95-.695 1.715-.798V49.6h.952v1.232c.476.019.866.063 1.169.133.303.07.576.161.819.273l-.336.98a3.584 3.584 0 00-.672-.245 4.736 4.736 0 00-.98-.133v3.458c.29.15.579.31.868.483.29.173.548.371.777.595.229.224.413.485.553.784.14.299.21.649.21 1.05 0 .728-.203 1.328-.609 1.799-.406.471-1.006.772-1.799.903V62.2h-.952v-1.232zm2.198-2.66c0-.224-.04-.425-.119-.602a1.724 1.724 0 00-.315-.476 2.327 2.327 0 00-.462-.378 6.612 6.612 0 00-.574-.322v3.402c.43-.075.782-.243 1.057-.504.275-.261.413-.635.413-1.12zm-3.318-5.054c0 .401.128.728.385.98s.576.48.959.686v-3.066c-.495.056-.842.22-1.043.49-.2.27-.301.574-.301.91zm8.918 7.098c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715v-1.372z"/><path id="name" fill="#181717" d="M193.382 56.25a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547V60.8h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665V60.8h-1.078v-4.55zm8.61-1.904a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm8.498.812v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98V60.8h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889V60.8h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12V60.8h-1.064zm12.138-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="}" fill="#7E7C7B" d="M231 61.724c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V60.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V58c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="`" fill="#478964" d="M239.26 52.806h-.77l-1.666-2.086h1.526z"/><path id=");}" fill="#7E7C7B" d="M252.882 62.802a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.724c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.03c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119V75c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><text id="execution-start" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="20" y="30">execution start</tspan></text><g id="Group-3" transform="translate(311)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h204v48H1z"/><text id="phrase:-<uninitializ" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="21">phrase: <uninitialized></tspan> <tspan x="8" y="36">say: function</tspan></text></g><g id="Group-3-Copy" opacity=".5" transform="translate(311 61)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h204v29H1z"/><text id="..." fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="88" y="16">...</tspan></text></g><path id="Line" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M244.5 26.5h50"/><path id="Line-Copy" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M244.5 77.5h50"/></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/closure-makecounter-environment.svg b/1-js/06-advanced-functions/03-closure/closure-makecounter-environment.svg new file mode 100644 index 000000000..f78441712 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-makecounter-environment.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="775" height="172" viewBox="0 0 775 172"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-makecounter-environment.svg"><g id="function-makeCounter" fill-rule="nonzero" transform="translate(5 12)"><path id="function" fill="#1C85B5" d="M.112 9.968V9.03h1.666V3.906H.112v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87V9.03h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm9.436 7V9.03h2.436V3.906H42.35v-.938h3.556V9.03h2.38v.938H42.35zM44.408.98c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="makeCounter" fill="#181717" d="M78.148 9.968V5.306c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM91.616.168h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H97.65l-2.996-3.164h-.77v3.164h-1.078V1.106h-1.19V.168zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014V2.73h-1.036V1.232zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244V9.03h1.89V3.906h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="(){" fill="#7E7C7B" d="M172.928 12.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.464c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.52z"/><path id="let" fill="#1C85B5" d="M17.346 17.168h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="count" fill="#181717" d="M55.426 21.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H72.31l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V23.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="=" fill="#DBAF88" d="M100.632 20.57h6.496v1.008h-6.496V20.57zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="0" fill="#A7333A" d="M117.25 22.068c0-.793.07-1.505.21-2.135.14-.63.35-1.162.63-1.596.28-.434.635-.765 1.064-.994.43-.229.938-.343 1.526-.343.625 0 1.155.112 1.589.336.434.224.786.55 1.057.98.27.43.467.959.588 1.589.121.63.182 1.351.182 2.163a9.88 9.88 0 01-.21 2.135c-.14.63-.35 1.162-.63 1.596-.28.434-.635.765-1.064.994-.43.229-.938.343-1.526.343-.616 0-1.141-.124-1.575-.371a2.956 2.956 0 01-1.064-1.043c-.275-.448-.474-.982-.595-1.603a10.713 10.713 0 01-.182-2.051zm5.698 0c0-.495-.028-.966-.084-1.414l-4.158 3.794c.159.532.397.957.714 1.274.317.317.733.476 1.246.476.821 0 1.407-.34 1.757-1.022.35-.681.525-1.717.525-3.108zm-4.536 0c0 .233.007.457.021.672.014.215.03.425.049.63l4.172-3.78c-.159-.504-.394-.905-.707-1.204-.313-.299-.735-.448-1.267-.448-.83 0-1.416.343-1.757 1.029-.34.686-.511 1.72-.511 3.101z"/><path id=";" fill="#7E7C7B" d="M128.086 26.198c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="returnfunction(){" fill="#C06334" d="M16.604 60.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm16.73 4.55v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H78.47v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H89.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.146 7.518a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V60.52z"/><path id="returncount++;" fill="#7E7C7B" d="M33.404 77.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H42.91c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H63.91l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm21.644-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm8.638 1.764h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm8.4 0h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm10.934 4.466c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="};" fill="#C06334" d="M21 95.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H16.8v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V87.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H16.8v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="}" fill="#7E7C7B" d="M4.2 112.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V104.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="letcounter=makeCounter();" fill="#C06334" d="M.546 136.168H2.87v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H.546v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H9.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm22.512 1.078a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H55.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H76.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm16.828-6.398h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008zm19.516 4.102v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.166-8.988h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014v2.212h-1.036v-1.498zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm13.524 2.968a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(329 28)"><path id="Line" d="M.709.967h16.789"/><path id="Line-Copy" d="M.709 86.033h16.789"/><path id="Line-Copy-2" d="M18.333.967v85.066"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(488 12)"><path id="Line-Copy-2" d="M2.333 1.7v149.6"/></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="741" y="71">null</tspan></text><text id="[[Environment]]" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="183" y="73">[[Environment]]</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M326.5 62.5l14 7-14 7v-6h-14v-2h14v-6z"/><path id="Rectangle-1-Copy-2" fill="#C06334" d="M0 75V62l9 6.5"/><g id="Group-3" transform="translate(498 44)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h187v48H1z"/><text id="makeCounter:-functio" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="6" y="21">makeCounter: function</tspan> <tspan x="6" y="36">counter: undefined</tspan></text></g><g id="Group-2" transform="translate(355 55)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h83v27H1z"/><text id="count:-0" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8.5" y="18">count: 0</tspan></text></g><g id="Group" transform="translate(445 50)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(695 48)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call-2.svg b/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call-2.svg new file mode 100644 index 000000000..3950a8faa --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call-2.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="774" height="222" viewBox="0 0 774 222"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-makecounter-nested-call-2.svg"><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(325 28)"><path id="Line" d="M.709.967h16.789"/><path id="Line-Copy" d="M.709 86.033h16.789"/><path id="Line-Copy-2" d="M18.333.967v85.066"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(480 25)"><path id="Line-Copy-2" d="M2.333 1.7v149.6"/></g><g id="Group-2" transform="translate(351 71)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v27H1z"/><text id="count:-1" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="4" y="19">count: 1</tspan></text></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="translate(207 67)"><path id="Line" d="M.355.422h8.394"/><path id="Line-Copy" d="M.355 37.578h8.394"/><path id="Line-Copy-2" d="M9.167.422v37.156"/></g><g id="Group-2-Copy" transform="translate(223 71)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h68v27H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="9.8" y="18"><empty></tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="736" y="91">null</tspan></text><g id="Group" transform="translate(300 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(437 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(690 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-3" transform="translate(493 60)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h187v48H1z"/><text id="makeCounter:-functio" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="6" y="21">makeCounter: function</tspan> <tspan x="6" y="36">counter: function</tspan></text></g><g id="function-makeCounter" fill-rule="nonzero" transform="translate(14 12)"><path id="function" fill="#1C85B5" d="M.182 9.968V9.03h1.666V3.906H.182v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.94V9.03h2.898v.938H.182zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.58l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.7v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm9.436 7V9.03h2.436V3.906H42.42v-.938h3.556V9.03h2.38v.938H42.42zM44.478.98c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="makeCounter" fill="#181717" d="M78.218 9.968V5.306c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM91.686.168h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H97.72l-2.996-3.164h-.77v3.164h-1.078V1.106h-1.19V.168zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014V2.73h-1.036V1.232zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244V9.03h1.89V3.906h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="(){" fill="#7E7C7B" d="M172.998 12.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.464c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.52z"/><path id="let" fill="#1C85B5" d="M17.416 17.168h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.18c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="count" fill="#181717" d="M55.496 21.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H72.38l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H66.5v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V23.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="=" fill="#DBAF88" d="M100.702 20.57h6.496v1.008h-6.496V20.57zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="0" fill="#A7333A" d="M117.32 22.068c0-.793.07-1.505.21-2.135.14-.63.35-1.162.63-1.596.28-.434.635-.765 1.064-.994.43-.229.938-.343 1.526-.343.625 0 1.155.112 1.589.336.434.224.786.55 1.057.98.27.43.467.959.588 1.589.121.63.182 1.351.182 2.163a9.88 9.88 0 01-.21 2.135c-.14.63-.35 1.162-.63 1.596-.28.434-.635.765-1.064.994-.43.229-.938.343-1.526.343-.616 0-1.141-.124-1.575-.371a2.956 2.956 0 01-1.064-1.043c-.275-.448-.474-.982-.595-1.603a10.713 10.713 0 01-.182-2.051zm5.698 0c0-.495-.028-.966-.084-1.414l-4.158 3.794c.159.532.397.957.714 1.274.317.317.733.476 1.246.476.821 0 1.407-.34 1.757-1.022.35-.681.525-1.717.525-3.108zm-4.536 0c0 .233.007.457.021.672.014.215.03.425.049.63l4.172-3.78c-.159-.504-.394-.905-.707-1.204-.313-.299-.735-.448-1.267-.448-.83 0-1.416.343-1.757 1.029-.34.686-.511 1.72-.511 3.101z"/><path id=";" fill="#7E7C7B" d="M128.156 26.198c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="returnfunction" fill="#1C85B5" d="M16.674 60.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.18c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.18l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H41.3v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm16.73 4.55v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H78.54v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H89.18l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H83.3v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M147.798 63.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V60.52z"/><path id="return" fill="#1C85B5" d="M33.474 77.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H42.98c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H63.98l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H58.1v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="count" fill="#181717" d="M97.496 72.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="++" fill="#DBAF88" d="M134.022 72.732h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm8.4 0h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008z"/><path id=";};}" fill="#7E7C7B" d="M153.356 77.198c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM21.07 95.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V87.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.27 112.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H.07v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V104.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H.07v-.98H2.1c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="let" fill="#1C85B5" d="M.616 136.168H2.94v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H.616v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H9.38c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="counter" fill="#181717" d="M38.696 140.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H55.58l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H49.7v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H76.58c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="=" fill="#DBAF88" d="M100.702 139.57h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="makeCounter" fill="#181717" d="M120.218 145.968v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.166-8.988h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014v2.212h-1.036v-1.498zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="();" fill="#7E7C7B" d="M214.998 148.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="alert" fill="#181717" d="M.462 173.514a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91H4.718l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm6.496-8.988h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H9.016v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H17.78c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="(" fill="#7E7C7B" d="M46.998 182.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966z"/><path id="counter" fill="#181717" d="M63.896 174.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H80.78l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H74.9v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="());//0" fill="#7E7C7B" d="M122.598 182.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm16.8 0a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM173.264 170l.882.392-5.11 11.536-.882-.392 5.11-11.536zm8.4 0l.882.392-5.11 11.536-.882-.392 5.11-11.536zm11.256 5.068c0-.793.07-1.505.21-2.135.14-.63.35-1.162.63-1.596.28-.434.635-.765 1.064-.994.43-.229.938-.343 1.526-.343.625 0 1.155.112 1.589.336.434.224.786.55 1.057.98.27.43.467.959.588 1.589.121.63.182 1.351.182 2.163a9.88 9.88 0 01-.21 2.135c-.14.63-.35 1.162-.63 1.596-.28.434-.635.765-1.064.994-.43.229-.938.343-1.526.343-.616 0-1.141-.124-1.575-.371a2.956 2.956 0 01-1.064-1.043c-.275-.448-.474-.982-.595-1.603a10.713 10.713 0 01-.182-2.051zm5.698 0c0-.495-.028-.966-.084-1.414l-4.158 3.794c.159.532.397.957.714 1.274.317.317.733.476 1.246.476.821 0 1.407-.34 1.757-1.022.35-.681.525-1.717.525-3.108zm-4.536 0c0 .233.007.457.021.672.014.215.03.425.049.63l4.172-3.78c-.159-.504-.394-.905-.707-1.204-.313-.299-.735-.448-1.267-.448-.83 0-1.416.343-1.757 1.029-.34.686-.511 1.72-.511 3.101z"/></g><text id="modified-here" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="356" y="62">modified here</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call.svg b/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call.svg new file mode 100644 index 000000000..24315bf21 --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-makecounter-nested-call.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="774" height="210" viewBox="0 0 774 210"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-makecounter-nested-call.svg"><g id="function-makeCounter" fill-rule="nonzero" transform="translate(21 12)"><path id="function" fill="#1C85B5" d="M.182 9.968V9.03h1.666V3.906H.182v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.94V9.03h2.898v.938H.182zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.58l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.7v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm9.436 7V9.03h2.436V3.906H42.42v-.938h3.556V9.03h2.38v.938H42.42zM44.478.98c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="makeCounter" fill="#181717" d="M78.218 9.968V5.306c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM91.686.168h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H97.72l-2.996-3.164h-.77v3.164h-1.078V1.106h-1.19V.168zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014V2.73h-1.036V1.232zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244V9.03h1.89V3.906h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="(){" fill="#7E7C7B" d="M172.998 12.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.464c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.52z"/><path id="let" fill="#1C85B5" d="M17.416 17.168h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.18c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="count" fill="#181717" d="M55.496 21.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H72.38l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H66.5v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V23.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="=" fill="#DBAF88" d="M100.702 20.57h6.496v1.008h-6.496V20.57zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="0" fill="#A7333A" d="M117.32 22.068c0-.793.07-1.505.21-2.135.14-.63.35-1.162.63-1.596.28-.434.635-.765 1.064-.994.43-.229.938-.343 1.526-.343.625 0 1.155.112 1.589.336.434.224.786.55 1.057.98.27.43.467.959.588 1.589.121.63.182 1.351.182 2.163a9.88 9.88 0 01-.21 2.135c-.14.63-.35 1.162-.63 1.596-.28.434-.635.765-1.064.994-.43.229-.938.343-1.526.343-.616 0-1.141-.124-1.575-.371a2.956 2.956 0 01-1.064-1.043c-.275-.448-.474-.982-.595-1.603a10.713 10.713 0 01-.182-2.051zm5.698 0c0-.495-.028-.966-.084-1.414l-4.158 3.794c.159.532.397.957.714 1.274.317.317.733.476 1.246.476.821 0 1.407-.34 1.757-1.022.35-.681.525-1.717.525-3.108zm-4.536 0c0 .233.007.457.021.672.014.215.03.425.049.63l4.172-3.78c-.159-.504-.394-.905-.707-1.204-.313-.299-.735-.448-1.267-.448-.83 0-1.416.343-1.757 1.029-.34.686-.511 1.72-.511 3.101z"/><path id=";" fill="#7E7C7B" d="M128.156 26.198c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="returnfunction" fill="#1C85B5" d="M16.674 60.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.18c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.18l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H41.3v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm16.73 4.55v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H78.54v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H89.18l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H83.3v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="(){" fill="#7E7C7B" d="M147.798 63.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V60.52z"/><path id="returncount++;" fill="#C06334" d="M33.474 77.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H42.98c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H63.98l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H58.1v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm21.644-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm8.638 1.764h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm8.4 0h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm10.934 4.466c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="};}" fill="#7E7C7B" d="M21.07 95.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581h-2.03v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V87.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42h-1.54v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.27 112.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H.07v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V104.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H.07v-.98H2.1c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="let" fill="#1C85B5" d="M.616 136.168H2.94v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H.616v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H9.38c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938z"/><path id="counter" fill="#181717" d="M38.696 140.046a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H55.58l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H49.7v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H76.58c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="=" fill="#DBAF88" d="M100.702 139.57h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id="makeCounter" fill="#181717" d="M120.218 145.968v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.166-8.988h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014v2.212h-1.036v-1.498zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="();" fill="#7E7C7B" d="M214.998 148.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="alert(counter());" fill="#C06334" d="M.462 173.514a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91H4.718l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm6.496-8.988h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H9.016v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H17.78c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm7.91-7h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm14.014 9.968a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm16.898-8.89a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H80.78l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H74.9v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm13.524 2.968a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm16.8 0a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(325 28)"><path id="Line" d="M.709.967h16.789"/><path id="Line-Copy" d="M.709 86.033h16.789"/><path id="Line-Copy-2" d="M18.333.967v85.066"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(480 25)"><path id="Line-Copy-2" d="M2.333 1.7v149.6"/></g><g id="Group-2" transform="translate(351 71)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h75v27H1z"/><text id="count:-0" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="4" y="19">count: 0</tspan></text></g><g id="Group-5-Copy-2" stroke="#C06334" stroke-linecap="square" transform="translate(207 67)"><path id="Line" d="M.355.422h8.394"/><path id="Line-Copy" d="M.355 37.578h8.394"/><path id="Line-Copy-2" d="M9.167.422v37.156"/></g><g id="Group-2-Copy" transform="translate(223 71)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h68v27H1z"/><text id="<empty>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="9.8" y="18"><empty></tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="736" y="91">null</tspan></text><path id="Rectangle-1-Copy-3" fill="#C06334" d="M5 93V80l9 6.5"/><g id="Group" transform="translate(300 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(437 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(690 68)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group-3" transform="translate(493 60)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h187v48H1z"/><text id="makeCounter:-functio" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="6" y="21">makeCounter: function</tspan> <tspan x="6" y="36">counter: function</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/closure-makecounter.svg b/1-js/06-advanced-functions/03-closure/closure-makecounter.svg new file mode 100644 index 000000000..2ca06455a --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-makecounter.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="711" height="173" viewBox="0 0 711 173"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-makecounter.svg"><g id="function-makeCounter" fill-rule="nonzero" transform="translate(26.12 12.032)"><path id="function" fill="#1C85B5" d="M.112 9.968V9.03h1.666V3.906H.112v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87V9.03h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm9.436 7V9.03h2.436V3.906H42.35v-.938h3.556V9.03h2.38v.938H42.35zM44.408.98c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="makeCounter" fill="#181717" d="M78.148 9.968V5.306c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM91.616.168h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938H97.65l-2.996-3.164h-.77v3.164h-1.078V1.106h-1.19V.168zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014V2.73h-1.036V1.232zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.906h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666V1.582l1.092-.308v1.694h3.738v.938h-3.738V7.21c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.906h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244V9.03h1.89V3.906h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656z"/><path id="(){" fill="#7E7C7B" d="M172.928 12.936a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.464c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.52z"/><path id="letcount=0;" fill="#C06334" d="M17.346 17.168h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm22.512 1.078a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H72.31l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V23.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm17.318.602h6.496v1.008h-6.496V20.57zm0 2.296h6.496v1.008h-6.496v-1.008zm16.618-.798c0-.793.07-1.505.21-2.135.14-.63.35-1.162.63-1.596.28-.434.635-.765 1.064-.994.43-.229.938-.343 1.526-.343.625 0 1.155.112 1.589.336.434.224.786.55 1.057.98.27.43.467.959.588 1.589.121.63.182 1.351.182 2.163a9.88 9.88 0 01-.21 2.135c-.14.63-.35 1.162-.63 1.596-.28.434-.635.765-1.064.994-.43.229-.938.343-1.526.343-.616 0-1.141-.124-1.575-.371a2.956 2.956 0 01-1.064-1.043c-.275-.448-.474-.982-.595-1.603a10.713 10.713 0 01-.182-2.051zm5.698 0c0-.495-.028-.966-.084-1.414l-4.158 3.794c.159.532.397.957.714 1.274.317.317.733.476 1.246.476.821 0 1.407-.34 1.757-1.022.35-.681.525-1.717.525-3.108zm-4.536 0c0 .233.007.457.021.672.014.215.03.425.049.63l4.172-3.78c-.159-.504-.394-.905-.707-1.204-.313-.299-.735-.448-1.267-.448-.83 0-1.416.343-1.757 1.029-.34.686-.511 1.72-.511 3.101zm9.674 4.13c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 60.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="function(){returncount++;};" fill="#7E7C7B" d="M75.712 60.968v-.938h1.666v-5.124h-1.666v-.938h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H78.47v5.124h2.898v.938h-5.656zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H89.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm9.436 7v-.938h2.436v-5.124h-2.436v-.938h3.556v6.062h2.38v.938h-5.936zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V57.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm13.146 7.518a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078v-1.428c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246v1.358c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V60.52zM33.404 77.968v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H42.91c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H63.91l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm21.644-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V74.16c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm8.638 1.764h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm8.4 0h3.024v-3.094h1.008v3.094h3.024v1.008h-3.024v3.094h-1.008V73.74h-3.024v-1.008zm10.934 4.466c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM21 95.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H16.8v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V87.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H16.8v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372zm6.286-1.694c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="}" fill="#7E7C7B" d="M4.2 112.892c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26v-1.344c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022V104.8c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/><path id="letcounter=makeCounter();" fill="#C06334" d="M.546 136.168H2.87v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51H.546v-.938zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H9.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm22.512 1.078a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm3.052 2.422c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H55.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H76.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm16.828-6.398h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008zm19.516 4.102v-4.662c0-.196-.007-.385-.021-.567a1.9 1.9 0 00-.098-.49.786.786 0 00-.224-.343.57.57 0 00-.385-.126c-.317 0-.586.13-.805.392-.22.261-.385.588-.497.98v4.816h-1.064v-7h.728l.21.854h.056c.093-.14.184-.273.273-.399.089-.126.191-.236.308-.329.117-.093.254-.166.413-.217.159-.051.36-.077.602-.077.14 0 .285.021.434.063.15.042.287.107.413.196.126.089.236.208.329.357.093.15.154.331.182.546.215-.364.455-.649.721-.854.266-.205.632-.308 1.099-.308.308 0 .558.051.749.154.191.103.34.25.448.441.107.191.182.42.224.686.042.266.063.562.063.889v4.998h-1.064v-4.76c0-.196-.01-.38-.028-.553a1.764 1.764 0 00-.105-.455.752.752 0 00-.21-.308.536.536 0 00-.357-.112c-.327 0-.602.13-.826.392-.224.261-.392.635-.504 1.12v4.676h-1.064zm6.244-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.166-8.988h2.268v5.796h.77l2.758-2.996h1.316l-3.094 3.318 2.548 2.744h.91v.938h-1.442l-2.996-3.164h-.77v3.164h-1.078v-8.862h-1.19v-.938zm15.47 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm10.36-2.492a2.466 2.466 0 00-.525-.147 4.192 4.192 0 00-.693-.049c-.383 0-.758.07-1.127.21-.369.14-.695.369-.98.686-.285.317-.513.733-.686 1.246-.173.513-.259 1.143-.259 1.89 0 .672.077 1.26.231 1.764.154.504.369.924.644 1.26.275.336.607.588.994.756.387.168.819.252 1.295.252.243 0 .469-.021.679-.063.21-.042.404-.1.581-.175v-1.4h1.036v1.89l.014.028a3.66 3.66 0 01-1.064.539c-.43.145-.952.217-1.568.217-.588 0-1.132-.1-1.631-.301a3.41 3.41 0 01-1.295-.917c-.364-.41-.646-.936-.847-1.575-.2-.64-.301-1.398-.301-2.275 0-.915.112-1.692.336-2.331.224-.64.527-1.162.91-1.568a3.452 3.452 0 011.337-.889 4.697 4.697 0 011.631-.28c.317 0 .593.012.826.035.233.023.439.056.616.098s.334.096.469.161c.135.065.268.14.399.224h.014v2.212h-1.036v-1.498zm2.786 5.236c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.522-3.5h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.62 2.45a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55zm7.532-2.45h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm13.524 2.968a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm9.604-2.772c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><g id="Group-3" transform="translate(434 40)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h187v48H1z"/><text id="makeCounter:-functio" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="6" y="21">makeCounter: function</tspan> <tspan x="6" y="36">counter: undefined</tspan></text></g><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(239 28)"><path id="Line" d="M.709.967h16.789"/><path id="Line-Copy" d="M.709 86.033h16.789"/><path id="Line-Copy-2" d="M18.333.967v85.066"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(425 8)"><path id="Line-Copy-2" d="M2.333 1.7v149.6"/></g><g id="Group-2" transform="translate(281 51)"><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h83v27H1z"/><text id="count:-0" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8.5" y="18">count: 0</tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="677" y="68">null</tspan></text><path id="Rectangle-1-Copy" fill="#C06334" d="M3 39.774V27l9 6.387"/><text id="global-LexicalEnviro" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="440" y="15">global LexicalEnvironment</tspan></text><text id="LexicalEnvironment-o" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="275" y="15">LexicalEnvironment</tspan> <tspan x="275" y="29">of makeCounter() call</tspan></text><g id="Group" transform="translate(376 46)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><g id="Group" transform="translate(629 45)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27 12l14 7-14 7-.001-6H-1v-2h27.999L27 12z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/closure-variable-phrase.svg b/1-js/06-advanced-functions/03-closure/closure-variable-phrase.svg new file mode 100644 index 000000000..b9bb12fff --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/closure-variable-phrase.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="550" height="150" viewBox="0 0 550 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="closure-variable-phrase.svg"><g id="Group-3" transform="translate(235 114)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h204v28H1z"/><text id="phrase:-"Bye"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8.598" y="18.51">phrase: "Bye"</tspan></text></g><g id="Group-3-Copy" transform="translate(235 79)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h204v28H1z"/><text id="phrase:-"Hello"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8.598" y="18.51">phrase: "Hello"</tspan></text></g><g id="Group-3-Copy-3" transform="translate(235 44)"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 1h204v28H1z"/><text id="phrase:-undefined" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7" y="19.51">phrase: undefined</tspan></text></g><g id="Group-3-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" transform="translate(235 9)"><path id="Rectangle-1" d="M1 1h204v28H1z"/></g><text id="phrase:-<uninitializ" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="242" y="28">phrase: <uninitialized></tspan></text><g id="Group" transform="translate(450 3)"><path id="Line" fill="#C06334" fill-rule="nonzero" d="M27.5 13.5l14 7-14 7v-6h-28v-2h28v-6z"/><text id="outer" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="0" y="11">outer</tspan></text></g><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="497" y="27">null</tspan></text><g id="let-phrase;" fill-rule="nonzero" transform="translate(19 53)"><path id="let" fill="#1C85B5" d="M1.372 0h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197V.938H1.372V0zm14.14 8.918c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155A4.767 4.767 0 018.988 6.3c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM16.94 2.8h1.666V1.414l1.092-.308V2.8h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.738H16.94V2.8z"/><path id="phrase" fill="#181717" d="M34.65 5.25a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364H33.6V2.8h1.862l.14.98h.07c.252-.327.572-.6.959-.819.387-.22.861-.329 1.421-.329.999 0 1.75.275 2.254.826.504.55.756 1.451.756 2.702 0 .588-.086 1.118-.259 1.589a3.34 3.34 0 01-.735 1.197 3.24 3.24 0 01-1.148.756 3.999 3.999 0 01-1.484.266c-.196 0-.371-.007-.525-.021a4.613 4.613 0 01-.42-.056 3.31 3.31 0 01-.371-.091 5.103 5.103 0 01-.392-.14v2.94H34.65V5.25zm3.22-1.596c-.28 0-.544.051-.791.154a2.36 2.36 0 00-1.127.966c-.121.205-.196.41-.224.616v3.192c.205.14.436.25.693.329.257.08.595.119 1.015.119.747 0 1.344-.245 1.792-.735.448-.49.672-1.206.672-2.149 0-.793-.156-1.407-.469-1.841-.313-.434-.833-.651-1.561-.651zM41.762 0h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V9.8h-1.092V5.978c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V9.8h-1.078V.938h-1.106V0zm9.268 9.8v-.938h1.89V3.738h-1.89V2.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688V9.8H51.03zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zM73.038 7.91a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm9.674 1.008c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id=";" fill="#7E7C7B" d="M86.912 9.03c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="phrase" fill="#181717" d="M1.05 39.25a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364H0v-.91h1.862l.14.98h.07c.252-.327.572-.6.959-.819.387-.22.861-.329 1.421-.329.999 0 1.75.275 2.254.826.504.55.756 1.451.756 2.702 0 .588-.086 1.118-.259 1.589a3.34 3.34 0 01-.735 1.197 3.24 3.24 0 01-1.148.756 3.999 3.999 0 01-1.484.266c-.196 0-.371-.007-.525-.021a4.613 4.613 0 01-.42-.056 3.31 3.31 0 01-.371-.091 5.103 5.103 0 01-.392-.14v2.94H1.05v-7.35zm3.22-1.596c-.28 0-.544.051-.791.154a2.36 2.36 0 00-1.127.966c-.121.205-.196.41-.224.616v3.192c.205.14.436.25.693.329.257.08.595.119 1.015.119.747 0 1.344-.245 1.792-.735.448-.49.672-1.206.672-2.149 0-.793-.156-1.407-.469-1.841-.313-.434-.833-.651-1.561-.651zM8.162 34h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V43.8h-1.092v-3.822c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V43.8H9.268v-8.862H8.162V34zm9.268 9.8v-.938h1.89v-5.124h-1.89V36.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938H17.43zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm10.962-1.078a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm9.674 1.008c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="=" fill="#DBAF88" d="M59.458 37.402h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id=""Hello"" fill="#478964" d="M80.332 34h1.232l-.434 4.242h-.798V34zm-2.898 0h1.232l-.434 4.242h-.798V34zm12.67 5.306h-4.396V43.8h-1.162V34h1.162v4.27h4.396V34h1.162v9.8h-1.162v-4.494zm9.408 3.612c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM102.172 34h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V34zm8.4 0h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232V34zm7.532 6.3c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm11.466-6.3h1.232l-.434 4.242h-.798V34zm-2.898 0h1.232l-.434 4.242h-.798V34z"/><path id=";" fill="#7E7C7B" d="M137.312 43.03c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="phrase" fill="#181717" d="M1.05 73.25a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364H0v-.91h1.862l.14.98h.07c.252-.327.572-.6.959-.819.387-.22.861-.329 1.421-.329.999 0 1.75.275 2.254.826.504.55.756 1.451.756 2.702 0 .588-.086 1.118-.259 1.589a3.34 3.34 0 01-.735 1.197 3.24 3.24 0 01-1.148.756 3.999 3.999 0 01-1.484.266c-.196 0-.371-.007-.525-.021a4.613 4.613 0 01-.42-.056 3.31 3.31 0 01-.371-.091 5.103 5.103 0 01-.392-.14v2.94H1.05v-7.35zm3.22-1.596c-.28 0-.544.051-.791.154a2.36 2.36 0 00-1.127.966c-.121.205-.196.41-.224.616v3.192c.205.14.436.25.693.329.257.08.595.119 1.015.119.747 0 1.344-.245 1.792-.735.448-.49.672-1.206.672-2.149 0-.793-.156-1.407-.469-1.841-.313-.434-.833-.651-1.561-.651zM8.162 68h2.184v3.668h.07c.29-.327.625-.581 1.008-.763.383-.182.859-.273 1.428-.273.448 0 .838.047 1.169.14.331.093.602.257.812.49.21.233.366.553.469.959.103.406.154.922.154 1.547V77.8h-1.092v-3.822c0-.401-.023-.751-.07-1.05a1.829 1.829 0 00-.273-.749c-.135-.2-.32-.352-.553-.455-.233-.103-.532-.154-.896-.154-.252 0-.5.044-.742.133-.243.089-.464.21-.665.364-.2.154-.373.34-.518.56-.145.22-.245.46-.301.721V77.8H9.268v-8.862H8.162V68zm9.268 9.8v-.938h1.89v-5.124h-1.89V70.8h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938H17.43zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm10.962-1.078a.796.796 0 00-.35-.693 2.872 2.872 0 00-.868-.392 16.056 16.056 0 00-1.127-.273 6.502 6.502 0 01-1.127-.336 2.525 2.525 0 01-.868-.567c-.233-.238-.35-.572-.35-1.001 0-.355.077-.658.231-.91a1.94 1.94 0 01.609-.623c.252-.163.546-.285.882-.364.336-.08.686-.119 1.05-.119.653 0 1.216.082 1.687.245.471.163.847.334 1.127.511l-.448.882a11.79 11.79 0 00-1.001-.483c-.36-.154-.81-.231-1.351-.231-.205 0-.408.021-.609.063-.2.042-.383.105-.546.189a1.088 1.088 0 00-.392.329.825.825 0 00-.147.497c0 .243.117.43.35.56.233.13.523.24.868.329.345.089.721.173 1.127.252.406.08.782.194 1.127.343.345.15.635.35.868.602.233.252.35.593.35 1.022 0 .644-.254 1.176-.763 1.596-.509.42-1.272.63-2.289.63-.308 0-.611-.028-.91-.084a5.52 5.52 0 01-.84-.224 5.054 5.054 0 01-.714-.315 3.002 3.002 0 01-.532-.357l.56-.91c.112.112.261.226.448.343a4.229 4.229 0 001.337.539 3.76 3.76 0 001.484.021c.22-.042.413-.107.581-.196a1.08 1.08 0 00.399-.35.914.914 0 00.147-.525zm9.674 1.008c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="=" fill="#AF6E24" d="M59.458 71.402h6.496v1.008h-6.496v-1.008zm0 2.296h6.496v1.008h-6.496v-1.008z"/><path id=""Bye"" fill="#478964" d="M80.332 68h1.232l-.434 4.242h-.798V68zm-2.898 0h1.232l-.434 4.242h-.798V68zm13.58 2.24a2.42 2.42 0 01-.448 1.393c-.15.21-.34.394-.574.553-.233.159-.509.28-.826.364v.056c.28.047.548.126.805.238a2.19 2.19 0 011.155 1.176c.121.28.182.611.182.994 0 .504-.105.94-.315 1.309-.21.369-.485.67-.826.903-.34.233-.73.406-1.169.518a5.411 5.411 0 01-1.344.168h-.56c-.215 0-.439-.007-.672-.021a17.67 17.67 0 01-.7-.056 4.007 4.007 0 01-.616-.105v-9.646c.345-.056.747-.103 1.204-.14.457-.037.957-.056 1.498-.056.364 0 .735.03 1.113.091.378.06.721.177 1.029.35.308.173.562.413.763.721.2.308.301.705.301 1.19zm-3.248 6.664c.299 0 .588-.037.868-.112.28-.075.527-.187.742-.336.215-.15.385-.336.511-.56.126-.224.189-.49.189-.798 0-.383-.077-.69-.231-.924a1.613 1.613 0 00-.602-.546 2.605 2.605 0 00-.833-.266 6.16 6.16 0 00-.924-.07h-1.218v3.514c.065.019.159.033.28.042.121.01.252.019.392.028a12.308 12.308 0 00.826.028zm-.77-4.592c.159 0 .35-.005.574-.014.224-.01.41-.023.56-.042a3.83 3.83 0 00.63-.273c.196-.107.371-.236.525-.385.154-.15.275-.317.364-.504.089-.187.133-.392.133-.616 0-.308-.058-.565-.175-.77a1.35 1.35 0 00-.469-.49 1.932 1.932 0 00-.672-.252 4.22 4.22 0 00-.77-.07c-.308 0-.59.007-.847.021a4.04 4.04 0 00-.581.063v3.332h.728zm9.464 4.41h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm11.452.196c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zM113.932 68h1.232l-.434 4.242h-.798V68zm-2.898 0h1.232l-.434 4.242h-.798V68z"/><path id=";" fill="#7E7C7B" d="M120.512 77.03c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/></g><text id="execution-start" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="20" y="28">execution start</tspan></text><path id="Line-Copy" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M166.5 58.5h50"/><path id="Line-Copy-2" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M166.5 92.5h50"/><path id="Line-Copy-3" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M166.5 126.5h50"/><path id="Line-Copy-4" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M166.5 126.5h50"/><path id="Line" stroke="#7E7C7B" stroke-dasharray="2,4,2,4" stroke-linecap="square" d="M166.5 24.5h50"/></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/03-closure/variable-scope-lookup.svg b/1-js/06-advanced-functions/03-closure/variable-scope-lookup.svg new file mode 100644 index 000000000..f1f1d3b1d --- /dev/null +++ b/1-js/06-advanced-functions/03-closure/variable-scope-lookup.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="374" height="191" viewBox="0 0 374 191"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="closure" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="variable-scope-lookup.svg"><text id="function" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="30">function</tspan></text><text id="Use" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="88" y="30">Use</tspan></text><text id="r" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="113" y="30">r</tspan></text><text id="(" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="122" y="30">(</tspan></text><text id="nam" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="130" y="30">nam</tspan></text><text id="e" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="155" y="30">e</tspan></text><text id=")" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="164" y="30">)</tspan></text><text id="{" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="180" y="30">{</tspan></text><text id="thi" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="29" y="62">thi</tspan></text><text id="s" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="54" y="62">s</tspan></text><text id="." fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="63" y="62">.</tspan></text><text id="sayHi" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="71" y="62">sayHi</tspan></text><text id="=" fill="#DBAF88" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="122" y="62">=</tspan></text><text id="functio" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="138" y="62">functio</tspan></text><text id="n" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="197" y="62">n</tspan></text><text id="(" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="206" y="62">(</tspan></text><text id=")" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="214" y="62">)</tspan></text><text id="{" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="231" y="62">{</tspan></text><text id="aler" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="46" y="77">aler</tspan></text><text id="t" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="80" y="77">t</tspan></text><text id="(" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="88" y="77">(</tspan></text><text id="nam" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="96" y="77">nam</tspan></text><text id="e" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="122" y="77">e</tspan></text><text id=")" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="130" y="77">)</tspan></text><text id=";" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="138" y="77">;</tspan></text><text id="}" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="29" y="93">}</tspan></text><text id=";" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="38" y="93">;</tspan></text><text id="}" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="124">}</tspan></text><text id="let" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="156">let</tspan></text><text id="user" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="46" y="156">user</tspan></text><text id="=" fill="#DBAF88" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="88" y="156">=</tspan></text><text id="new" fill="#1C85B5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="105" y="156">new</tspan></text><text id="Use" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="138" y="156">Use</tspan></text><text id="r" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="164" y="156">r</tspan></text><text id="(" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="172" y="156">(</tspan></text><text id=""John" fill="#478964" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="180" y="156">"John</tspan></text><text id=""" fill="#478964" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="222" y="156">"</tspan></text><text id=")" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="231" y="156">)</tspan></text><text id=";" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="239" y="156">;</tspan></text><text id="use" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="171">use</tspan></text><text id="r" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="38" y="171">r</tspan></text><text id="." fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="46" y="171">.</tspan></text><text id="sayH" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="54" y="171">sayH</tspan></text><text id="i" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="88" y="171">i</tspan></text><text id="(" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="96" y="171">(</tspan></text><text id=")" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="105" y="171">)</tspan></text><text id=";" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="113" y="171">;</tspan></text><g id="Group-5" stroke="#C06334" stroke-linecap="square" transform="translate(247 49)"><path id="Line" d="M.129.5h7.742"/><path id="Line-Copy" d="M.129 55.5h7.742"/><path id="Line-Copy-2" d="M8.5.622v54.756"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="translate(293 17)"><path id="Line" d="M.129.5h7.742"/><path id="Line-Copy" d="M.129 109.5h7.742"/><path id="Line-Copy-2" d="M8.5 1.222v107.556"/></g><g id="Group-5-Copy" stroke="#C06334" stroke-linecap="square" transform="matrix(-1 0 0 1 345 3)"><path id="Line-Copy-2" d="M3 2.078v182.844"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M279.5 69.5l14 7-14 7v-6h-14v-2h14v-6zM322.5 69.5l14 7-14 7v-6h-14v-2h14v-6z"/></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/04-var/article.md b/1-js/06-advanced-functions/04-var/article.md index 0a3666641..28d7a76ec 100644 --- a/1-js/06-advanced-functions/04-var/article.md +++ b/1-js/06-advanced-functions/04-var/article.md @@ -1,41 +1,38 @@ # The old "var" +```smart header="This article is for understanding old scripts" +The information in this article is useful for understanding old scripts. + +That's not how we write new code. +``` + In the very first chapter about [variables](info:variables), we mentioned three ways of variable declaration: 1. `let` 2. `const` 3. `var` -`let` and `const` behave exactly the same way in terms of Lexical Environments. - -But `var` is a very different beast, that originates from very old times. It's generally not used in modern scripts, but still lurks in the old ones. - -If you don't plan on meeting such scripts you may even skip this chapter or postpone it, but then there's a chance that it bites you later. - -From the first sight, `var` behaves similar to `let`. That is, declares a variable: +The `var` declaration is similar to `let`. Most of the time we can replace `let` by `var` or vice-versa and expect things to work: ```js run -function sayHi() { - var phrase = "Hello"; // local variable, "var" instead of "let" +var message = "Hi"; +alert(message); // Hi +``` - alert(phrase); // Hello -} +But internally `var` is a very different beast, that originates from very old times. It's generally not used in modern scripts, but still lurks in the old ones. -sayHi(); +If you don't plan on meeting such scripts you may even skip this chapter or postpone it. -alert(phrase); // Error, phrase is not defined -``` - -...But here are the differences. +On the other hand, it's important to understand differences when migrating old scripts from `var` to `let`, to avoid odd errors. ## "var" has no block scope -`var` variables are either function-wide or global, they are visible through blocks. +Variables, declared with `var`, are either function-scoped or global-scoped. They are visible through blocks. For instance: -```js +```js run if (true) { var test = true; // use "var" instead of "let" } @@ -45,23 +42,37 @@ alert(test); // true, the variable lives after if */!* ``` -If we used `let test` on the 2nd line, then it wouldn't be visible to `alert`. But `var` ignores code blocks, so we've got a global `test`. +As `var` ignores code blocks, we've got a global variable `test`. + +If we used `let test` instead of `var test`, then the variable would only be visible inside `if`: + +```js run +if (true) { + let test = true; // use "let" +} + +*!* +alert(test); // ReferenceError: test is not defined +*/!* +``` The same thing for loops: `var` cannot be block- or loop-local: -```js +```js run for (var i = 0; i < 10; i++) { + var one = 1; // ... } *!* -alert(i); // 10, "i" is visible after loop, it's a global variable +alert(i); // 10, "i" is visible after loop, it's a global variable +alert(one); // 1, "one" is visible after loop, it's a global variable */!* ``` If a code block is inside a function, then `var` becomes a function-level variable: -```js +```js run function sayHi() { if (true) { var phrase = "Hello"; @@ -71,12 +82,32 @@ function sayHi() { } sayHi(); -alert(phrase); // Error: phrase is not defined +alert(phrase); // ReferenceError: phrase is not defined ``` -As we can see, `var` pierces through `if`, `for` or other code blocks. That's because a long time ago in JavaScript blocks had no Lexical Environments. And `var` is a remnant of that. +As we can see, `var` pierces through `if`, `for` or other code blocks. That's because a long time ago in JavaScript, blocks had no Lexical Environments, and `var` is a remnant of that. + +## "var" tolerates redeclarations + +If we declare the same variable with `let` twice in the same scope, that's an error: + +```js run +let user; +let user; // SyntaxError: 'user' has already been declared +``` -## "var" are processed at the function start +With `var`, we can redeclare a variable any number of times. If we use `var` with an already-declared variable, it's just ignored: + +```js run +var user = "Pete"; + +var user = "John"; // this "var" does nothing (already declared) +// ...it doesn't trigger an error + +alert(user); // John +``` + +## "var" variables can be declared below their use `var` declarations are processed when the function starts (or script starts for globals). @@ -84,7 +115,7 @@ In other words, `var` variables are defined from the beginning of the function, So this code: -```js +```js run function sayHi() { phrase = "Hello"; @@ -94,11 +125,12 @@ function sayHi() { var phrase; */!* } +sayHi(); ``` ...Is technically the same as this (moved `var phrase` above): -```js +```js run function sayHi() { *!* var phrase; @@ -108,11 +140,12 @@ function sayHi() { alert(phrase); } +sayHi(); ``` ...Or even as this (remember, code blocks are ignored): -```js +```js run function sayHi() { phrase = "Hello"; // (*) @@ -124,6 +157,7 @@ function sayHi() { alert(phrase); } +sayHi(); ``` People also call such behavior "hoisting" (raising), because all `var` are "hoisted" (raised) to the top of the function. @@ -132,11 +166,11 @@ So in the example above, `if (false)` branch never executes, but that doesn't ma **Declarations are hoisted, but assignments are not.** -That's better to demonstrate with an example, like this: +That's best demonstrated with an example: ```js run function sayHi() { - alert(phrase); + alert(phrase); *!* var phrase = "Hello"; @@ -171,15 +205,83 @@ sayHi(); Because all `var` declarations are processed at the function start, we can reference them at any place. But variables are undefined until the assignments. -In both examples above `alert` runs without an error, because the variable `phrase` exists. But its value is not yet assigned, so it shows `undefined`. +In both examples above, `alert` runs without an error, because the variable `phrase` exists. But its value is not yet assigned, so it shows `undefined`. + +## IIFE + +In the past, as there was only `var`, and it has no block-level visibility, programmers invented a way to emulate it. What they did was called "immediately-invoked function expressions" (abbreviated as IIFE). + +That's not something we should use nowadays, but you can find them in old scripts. + +An IIFE looks like this: + +```js run +(function() { + + var message = "Hello"; + + alert(message); // Hello + +})(); +``` + +Here, a Function Expression is created and immediately called. So the code executes right away and has its own private variables. + +The Function Expression is wrapped with parenthesis `(function {...})`, because when JavaScript engine encounters `"function"` in the main code, it understands it as the start of a Function Declaration. But a Function Declaration must have a name, so this kind of code will give an error: + +```js run +// Tries to declare and immediately call a function +function() { // <-- SyntaxError: Function statements require a function name + + var message = "Hello"; + + alert(message); // Hello + +}(); +``` + +Even if we say: "okay, let's add a name", that won't work, as JavaScript does not allow Function Declarations to be called immediately: + +```js run +// syntax error because of parentheses below +function go() { + +}(); // <-- can't call Function Declaration immediately +``` + +So, the parentheses around the function is a trick to show JavaScript that the function is created in the context of another expression, and hence it's a Function Expression: it needs no name and can be called immediately. + +There exist other ways besides parentheses to tell JavaScript that we mean a Function Expression: + +```js run +// Ways to create IIFE + +*!*(*/!*function() { + alert("Parentheses around the function"); +}*!*)*/!*(); + +*!*(*/!*function() { + alert("Parentheses around the whole thing"); +}()*!*)*/!*; + +*!*!*/!*function() { + alert("Bitwise NOT operator starts the expression"); +}(); + +*!*+*/!*function() { + alert("Unary plus starts the expression"); +}(); +``` + +In all the above cases we declare a Function Expression and run it immediately. Let's note again: nowadays there's no reason to write such code. ## Summary -There are two main differences of `var`: +There are two main differences of `var` compared to `let/const`: -1. Variables have no block scope, they are visible minimum at the function level. -2. Variable declarations are processed at function start. +1. `var` variables have no block scope, their visibility is scoped to current function, or global, if declared outside function. +2. `var` declarations are processed at function start (script start for globals). -There's one more minor difference related to the global object, we'll cover that in the next chapter. +There's one more very minor difference related to the global object, that we'll cover in the next chapter. -These differences are actually a bad thing most of the time. Block-level variables is such a great thing. That's why `let` was introduced in the standard long ago, and is now a major way (along with `const`) to declare a variable. +These differences make `var` worse than `let` most of the time. Block-level variables is such a great thing. That's why `let` was introduced in the standard long ago, and is now a major way (along with `const`) to declare a variable. diff --git a/1-js/06-advanced-functions/05-global-object/article.md b/1-js/06-advanced-functions/05-global-object/article.md index da4adc2b6..cf4839d94 100644 --- a/1-js/06-advanced-functions/05-global-object/article.md +++ b/1-js/06-advanced-functions/05-global-object/article.md @@ -1,155 +1,89 @@ # Global object -The global object provides variables and functions that are available anywhere. Mostly, the ones that are built into the language or the host environment. +The global object provides variables and functions that are available anywhere. By default, those that are built into the language or the environment. -In a browser it is named "window", for Node.js it is "global", for other environments it may have another name. +In a browser it is named `window`, for Node.js it is `global`, for other environments it may have another name. -For instance, we can call `alert` as a method of `window`: +Recently, `globalThis` was added to the language, as a standardized name for a global object, that should be supported across all environments. It's supported in all major browsers. + +We'll use `window` here, assuming that our environment is a browser. If your script may run in other environments, it's better to use `globalThis` instead. + +All properties of the global object can be accessed directly: ```js run alert("Hello"); - -// the same as +// is the same as window.alert("Hello"); ``` -We can reference other built-in functions like `Array` as `window.Array` and create our own properties on it. - -## Browser: the "window" object - -For historical reasons, in-browser `window` object is a bit messed up. - -1. It provides the "browser window" functionality, besides playing the role of a global object. - - We can use `window` to access properties and methods, specific to the browser window: - - ```js run - alert(window.innerHeight); // shows the browser window height - - window.open('http://google.com'); // opens a new browser window - ``` - -2. Top-level `var` variables and function declarations automatically become properties of `window`. - - For instance: - ```js untrusted run no-strict refresh - var x = 5; - - alert(window.x); // 5 (var x becomes a property of window) - - window.x = 0; - - alert(x); // 0, variable modified - ``` - - Please note, that doesn't happen with more modern `let/const` declarations: - - ```js untrusted run no-strict refresh - let x = 5; - - alert(window.x); // undefined ("let" doesn't create a window property) - ``` - -3. Also, all scripts share the same global scope, so variables declared in one `<script>` become visible in another ones: - - ```html run - <script> - var a = 1; - let b = 2; - </script> +In a browser, global functions and variables declared with `var` (not `let/const`!) become the property of the global object: - <script> - alert(a); // 1 - alert(b); // 2 - </script> - ``` +```js run untrusted refresh +var gVar = 5; -4. And, a minor thing, but still: the value of `this` in the global scope is `window`. - - ```js untrusted run no-strict refresh - alert(this); // window - ``` - -Why was it made like this? At the time of the language creation, the idea to merge multiple aspects into a single `window` object was to "make things simple". But since then many things changed. Tiny scripts became big applications that require proper architecture. - -Is it good that different scripts (possibly from different sources) see variables of each other? - -No, it's not, because it may lead to naming conflicts: the same variable name can be used in two scripts for different purposes, so they will conflict with each other. - -As of now, the multi-purpose `window` is considered a design mistake in the language. - -Luckily, there's a "road out of hell", called "JavaScript modules". - -If we set `type="module"` attribute on a `<script>` tag, then such script is considered a separate "module" with its own top-level scope (lexical environment), not interfering with `window`. - -- In a module, `var x` does not become a property of `window`: - - ```html run - <script type="module"> - var x = 5; - - alert(window.x); // undefined - </script> - ``` +alert(window.gVar); // 5 (became a property of the global object) +``` -- Two modules that do not see variables of each other: +Function declarations have the same effect (statements with `function` keyword in the main code flow, not function expressions). - ```html run - <script type="module"> - let x = 5; - </script> +Please don't rely on that! This behavior exists for compatibility reasons. Modern scripts use [JavaScript modules](info:modules) where such a thing doesn't happen. - <script type="module"> - alert(window.x); // undefined - alert(x); // Error: undeclared variable - </script> - ``` +If we used `let` instead, such thing wouldn't happen: -- And, the last minor thing, the top-level value of `this` in a module is `undefined` (why should it be `window` anyway?): +```js run untrusted refresh +let gLet = 5; - ```html run - <script type="module"> - alert(this); // undefined - </script> - ``` +alert(window.gLet); // undefined (doesn't become a property of the global object) +``` -**Using `<script type="module">` fixes the design flaw of the language by separating top-level scope from `window`.** +If a value is so important that you'd like to make it available globally, write it directly as a property: -We'll cover more features of modules later, in the chapter [](info:modules). +```js run +*!* +// make current user information global, to let all scripts access it +window.currentUser = { + name: "John" +}; +*/!* + +// somewhere else in code +alert(currentUser.name); // John + +// or, if we have a local variable with the name "currentUser" +// get it from window explicitly (safe!) +alert(window.currentUser.name); // John +``` -## Valid uses of the global object +That said, using global variables is generally discouraged. There should be as few global variables as possible. The code design where a function gets "input" variables and produces certain "outcome" is clearer, less prone to errors and easier to test than if it uses outer or global variables. -1. Using global variables is generally discouraged. There should be as few global variables as possible, but if we need to make something globally visible, we may want to put it into `window` (or `global` in Node.js). +## Using for polyfills - Here we put the information about the current user into a global object, to be accessible from all other scripts: +We use the global object to test for support of modern language features. - ```js run - // explicitly assign it to `window` - window.currentUser = { - name: "John", - age: 30 - }; +For instance, test if a built-in `Promise` object exists (it doesn't in really old browsers): +```js run +if (!window.Promise) { + alert("Your browser is really old!"); +} +``` - // then, elsewhere, in another script - alert(window.currentUser.name); // John - ``` +If there's none (say, we're in an old browser), we can create "polyfills": add functions that are not supported by the environment, but exist in the modern standard. -2. We can test the global object for support of modern language features. +```js run +if (!window.Promise) { + window.Promise = ... // custom implementation of the modern language feature +} +``` - For instance, test if a build-in `Promise` object exists (it doesn't in really old browsers): - ```js run - if (!window.Promise) { - alert("Your browser is really old!"); - } - ``` +## Summary -3. We can create "polyfills": add functions that are not supported by the environment (say, an old browser), but exist in the modern standard. +- The global object holds variables that should be available everywhere. - ```js run - if (!window.Promise) { - window.Promise = ... // custom implementation of the modern language feature - } - ``` + That includes JavaScript built-ins, such as `Array` and environment-specific values, such as `window.innerHeight` -- the window height in the browser. +- The global object has a universal name `globalThis`. -...And of course, if we're in a browser, using `window` to access browser window features (not as a global object) is completely fine. + ...But more often is referred by "old-school" environment-specific names, such as `window` (browser) and `global` (Node.js). +- We should store values in the global object only if they're truly global for our project. And keep their number at minimum. +- In-browser, unless we're using [modules](info:modules), global functions and variables declared with `var` become a property of the global object. +- To make our code future-proof and easier to understand, we should access properties of the global object directly, as `window.x`. diff --git a/1-js/06-advanced-functions/06-function-object/2-counter-inc-dec/task.md b/1-js/06-advanced-functions/06-function-object/2-counter-inc-dec/task.md index 0177c8f6e..a11821d67 100644 --- a/1-js/06-advanced-functions/06-function-object/2-counter-inc-dec/task.md +++ b/1-js/06-advanced-functions/06-function-object/2-counter-inc-dec/task.md @@ -7,8 +7,8 @@ importance: 5 Modify the code of `makeCounter()` so that the counter can also decrease and set the number: - `counter()` should return the next number (as before). -- `counter.set(value)` should set the `count` to `value`. -- `counter.decrease()` should decrease the `count` by 1. +- `counter.set(value)` should set the counter to `value`. +- `counter.decrease()` should decrease the counter by 1. See the sandbox code for the complete usage example. diff --git a/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/solution.js b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/solution.js new file mode 100644 index 000000000..c7d7d734e --- /dev/null +++ b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/solution.js @@ -0,0 +1,15 @@ +function sum(a) { + + let currentSum = a; + + function f(b) { + currentSum += b; + return f; + } + + f.toString = function() { + return currentSum; + }; + + return f; +} diff --git a/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/source.js b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/source.js new file mode 100644 index 000000000..f10dca5dc --- /dev/null +++ b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/source.js @@ -0,0 +1,12 @@ +function sum(a){ + // Your code goes here. + +} + +/* +sum(1)(2) == 3; // 1 + 2 +sum(1)(2)(3) == 6; // 1 + 2 + 3 +sum(5)(-1)(2) == 6 +sum(6)(-1)(-2)(-3) == 0 +sum(0)(1)(2)(3)(4)(5) == 15 +*/ diff --git a/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/test.js b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/test.js new file mode 100644 index 000000000..ed567d330 --- /dev/null +++ b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/_js.view/test.js @@ -0,0 +1,19 @@ +describe("sum", function(){ + + it("sum(1)(2) == 3", function(){ + assert.equal(3, sum(1)(2)); + }); + + it("sum(5)(-1)(2) == 6", function(){ + assert.equal(6, sum(5)(-1)(2)); + }); + + it("sum(6)(-1)(-2)(-3) == 0", function(){ + assert.equal(0, sum(6)(-1)(-2)(-3)); + }); + + it("sum(0)(1)(2)(3)(4)(5) == 15", function(){ + assert.equal(15, sum(0)(1)(2)(3)(4)(5)); + }); +}); + diff --git a/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/solution.md b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/solution.md index 5c9326912..e97039f72 100644 --- a/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/solution.md +++ b/1-js/06-advanced-functions/06-function-object/5-sum-many-brackets/solution.md @@ -5,7 +5,7 @@ Now the code: -```js run +```js demo run function sum(a) { let currentSum = a; @@ -52,4 +52,4 @@ function f(b) { } ``` -This `f` will be used in the next call, again return itself, so many times as needed. Then, when used as a number or a string -- the `toString` returns the `currentSum`. We could also use `Symbol.toPrimitive` or `valueOf` here for the conversion. +This `f` will be used in the next call, again return itself, as many times as needed. Then, when used as a number or a string -- the `toString` returns the `currentSum`. We could also use `Symbol.toPrimitive` or `valueOf` here for the conversion. diff --git a/1-js/06-advanced-functions/06-function-object/article.md b/1-js/06-advanced-functions/06-function-object/article.md index 37839779c..8419ae763 100644 --- a/1-js/06-advanced-functions/06-function-object/article.md +++ b/1-js/06-advanced-functions/06-function-object/article.md @@ -1,7 +1,7 @@ # Function object, NFE -As we already know, functions in JavaScript are values. +As we already know, a function in JavaScript is a value. Every value in JavaScript has a type. What type is a function? @@ -12,7 +12,7 @@ A good way to imagine functions is as callable "action objects". We can not only ## The "name" property -Function objects contain a few useable properties. +Function objects contain some useable properties. For instance, a function's name is accessible as the "name" property: @@ -24,14 +24,14 @@ function sayHi() { alert(sayHi.name); // sayHi ``` -What's more funny, the name-assigning logic is smart. It also assigns the correct name to functions that are used in assignments: +What's kind of funny, the name-assigning logic is smart. It also assigns the correct name to a function even if it's created without one, and then immediately assigned: ```js run let sayHi = function() { alert("Hi"); -} +}; -alert(sayHi.name); // sayHi (works!) +alert(sayHi.name); // sayHi (there's a name!) ``` It also works if the assignment is done via a default value: @@ -67,7 +67,7 @@ alert(user.sayBye.name); // sayBye There's no magic though. There are cases when there's no way to figure out the right name. In that case, the name property is empty, like here: -```js +```js run // function created inside array let arr = [function() {}]; @@ -93,7 +93,7 @@ alert(many.length); // 2 Here we can see that rest parameters are not counted. -The `length` property is sometimes used for introspection in functions that operate on other functions. +The `length` property is sometimes used for [introspection](https://en.wikipedia.org/wiki/Type_introspection) in functions that operate on other functions. For instance, in the code below the `ask` function accepts a `question` to ask and an arbitrary number of `handler` functions to call. @@ -102,9 +102,9 @@ Once a user provides their answer, the function calls the handlers. We can pass - A zero-argument function, which is only called when the user gives a positive answer. - A function with arguments, which is called in either case and returns an answer. -The idea is that we have a simple, no-arguments handler syntax for positive cases (most frequent variant), but are able to provide universal handlers as well. +To call `handler` the right way, we examine the `handler.length` property. -To call `handlers` the right way, we examine the `length` property: +The idea is that we have a simple, no-arguments handler syntax for positive cases (most frequent variant), but are able to support universal handlers as well: ```js run function ask(question, ...handlers) { @@ -153,7 +153,7 @@ alert( `Called ${sayHi.counter} times` ); // Called 2 times ```warn header="A property is not a variable" A property assigned to a function like `sayHi.counter = 0` does *not* define a local variable `counter` inside it. In other words, a property `counter` and a variable `let counter` are two unrelated things. -We can treat a function as an object, store properties in it, but that has no effect on its execution. Variables never use function properties and vice versa. These are just parallel worlds. +We can treat a function as an object, store properties in it, but that has no effect on its execution. Variables are not function properties and vice versa. These are just parallel worlds. ``` Function properties can replace closures sometimes. For instance, we can rewrite the counter function example from the chapter <info:closure> to use a function property: @@ -241,7 +241,7 @@ let sayHi = function *!*func*/!*(who) { sayHi("John"); // Hello, John ``` -There are two special things about the name `func`: +There are two special things about the name `func`, that are the reasons for it: 1. It allows the function to reference itself internally. 2. It is not visible outside of the function. @@ -282,7 +282,7 @@ let sayHi = function(who) { }; ``` -The problem with that code is that the value of `sayHi` may change. The function may go to another variable, and the code will start to give errors: +The problem with that code is that `sayHi` may change in the outer code. If the function gets assigned to another variable instead, the code will start to give errors: ```js run let sayHi = function(who) { @@ -326,10 +326,10 @@ welcome(); // Hello, Guest (nested call works) Now it works, because the name `"func"` is function-local. It is not taken from outside (and not visible there). The specification guarantees that it will always reference the current function. -The outer code still has it's variable `sayHi` or `welcome`. And `func` is an "internal function name", how the function can call itself internally. +The outer code still has its variable `sayHi` or `welcome`. And `func` is an "internal function name", the way for the function to call itself reliably. ```smart header="There's no such thing for Function Declaration" -The "internal name" feature described here is only available for Function Expressions, not to Function Declarations. For Function Declarations, there's just no syntax possibility to add a one more "internal" name. +The "internal name" feature described here is only available for Function Expressions, not for Function Declarations. For Function Declarations, there is no syntax for adding an "internal" name. Sometimes, when we need a reliable internal name, it's the reason to rewrite a Function Declaration to Named Function Expression form. ``` @@ -340,13 +340,14 @@ Functions are objects. Here we covered their properties: -- `name` -- the function name. Exists not only when given in the function definition, but also for assignments and object properties. +- `name` -- the function name. Usually taken from the function definition, but if there's none, JavaScript tries to guess it from the context (e.g. an assignment). - `length` -- the number of arguments in the function definition. Rest parameters are not counted. If the function is declared as a Function Expression (not in the main code flow), and it carries the name, then it is called a Named Function Expression. The name can be used inside to reference itself, for recursive calls or such. Also, functions may carry additional properties. Many well-known JavaScript libraries make great use of this feature. -They create a "main" function and attach many other "helper" functions to it. For instance, the [jquery](https://jquery.com) library creates a function named `$`. The [lodash](https://lodash.com) library creates a function `_`. And then adds `_.clone`, `_.keyBy` and other properties to (see the [docs](https://lodash.com/docs) when you want learn more about them). Actually, they do it to lessen their pollution of the global space, so that a single library gives only one global variable. That reduces the possibility of naming conflicts. +They create a "main" function and attach many other "helper" functions to it. For instance, the [jQuery](https://jquery.com) library creates a function named `$`. The [lodash](https://lodash.com) library creates a function `_`, and then adds `_.clone`, `_.keyBy` and other properties to it (see the [docs](https://lodash.com/docs) when you want to learn more about them). Actually, they do it to lessen their pollution of the global space, so that a single library gives only one global variable. That reduces the possibility of naming conflicts. + So, a function can do a useful job by itself and also carry a bunch of other functionality in properties. diff --git a/1-js/06-advanced-functions/07-new-function/article.md b/1-js/06-advanced-functions/07-new-function/article.md index e0750e321..ffe264a4e 100644 --- a/1-js/06-advanced-functions/07-new-function/article.md +++ b/1-js/06-advanced-functions/07-new-function/article.md @@ -8,20 +8,20 @@ There's one more way to create a function. It's rarely used, but sometimes there The syntax for creating a function: ```js -let func = new Function ([arg1[, arg2[, ...argN]],] functionBody) +let func = new Function ([arg1, arg2, ...argN], functionBody); ``` -In other words, function parameters (or, more precisely, names for them) go first, and the body is last. All arguments are strings. +The function is created with the arguments `arg1...argN` and the given `functionBody`. It's easier to understand by looking at an example. Here's a function with two arguments: ```js run -let sum = new Function('a', 'b', 'return a + b'); +let sum = new Function('a', 'b', 'return a + b'); alert( sum(1, 2) ); // 3 ``` -If there are no arguments, then there's only a single argument, the function body: +And here there's a function without arguments, with only the function body: ```js run let sayHi = new Function('alert("Hello")'); @@ -29,7 +29,7 @@ let sayHi = new Function('alert("Hello")'); sayHi(); // Hello ``` -The major difference from other ways we've seen is that the function is created literally from a string, that is passed at run time. +The major difference from other ways we've seen is that the function is created literally from a string, that is passed at run time. All previous declarations required us, programmers, to write the function code in the script. @@ -42,16 +42,17 @@ let func = new Function(str); func(); ``` -It is used in very specific cases, like when we receive code from a server, or to dynamically compile a function from a template. The need for that usually arises at advanced stages of development. +It is used in very specific cases, like when we receive code from a server, or to dynamically compile a function from a template, in complex web-applications. ## Closure -Usually, a function remembers where it was born in the special property `[[Environment]]`. It references the Lexical Environment from where it's created. +Usually, a function remembers where it was born in the special property `[[Environment]]`. It references the Lexical Environment from where it's created (we covered that in the chapter <info:closure>). -But when a function is created using `new Function`, its `[[Environment]]` references not the current Lexical Environment, but instead the global one. +But when a function is created using `new Function`, its `[[Environment]]` is set to reference not the current Lexical Environment, but the global one. -```js run +So, such function doesn't have access to outer variables, only to the global ones. +```js run function getFunc() { let value = "test"; @@ -67,7 +68,7 @@ getFunc()(); // error: value is not defined Compare it with the regular behavior: -```js run +```js run function getFunc() { let value = "test"; @@ -87,51 +88,36 @@ Imagine that we must create a function from a string. The code of that function Our new function needs to interact with the main script. -Perhaps we want it to be able to access outer local variables? +What if it could access the outer variables? The problem is that before JavaScript is published to production, it's compressed using a *minifier* -- a special program that shrinks code by removing extra comments, spaces and -- what's important, renames local variables into shorter ones. -For instance, if a function has `let userName`, minifier replaces it `let a` (or another letter if this one is occupied), and does it everywhere. That's usually a safe thing to do, because the variable is local, nothing outside the function can access it. And inside the function, minifier replaces every mention of it. Minifiers are smart, they analyze the code structure, so they don't break anything. They're not just a dumb find-and-replace. +For instance, if a function has `let userName`, minifier replaces it with `let a` (or another letter if this one is occupied), and does it everywhere. That's usually a safe thing to do, because the variable is local, nothing outside the function can access it. And inside the function, minifier replaces every mention of it. Minifiers are smart, they analyze the code structure, so they don't break anything. They're not just a dumb find-and-replace. -But, if `new Function` could access outer variables, then it would be unable to find `userName`, since this is passed in as a string *after* the code is minified. +So if `new Function` had access to outer variables, it would be unable to find renamed `userName`. -**Even if we could access outer lexical environment in `new Function`, we would have problems with minifiers.** +**If `new Function` had access to outer variables, it would have problems with minifiers.** -The "special feature" of `new Function` saves us from mistakes. +Besides, such code would be architecturally bad and prone to errors. -And it enforces better code. If we need to pass something to a function created by `new Function`, we should pass it explicitly as an argument. - -Our "sum" function actually does that right: - -```js run -*!* -let sum = new Function('a', 'b', 'return a + b'); -*/!* - -let a = 1, b = 2; - -*!* -// outer values are passed as arguments -alert( sum(a, b) ); // 3 -*/!* -``` +To pass something to a function, created as `new Function`, we should use its arguments. ## Summary The syntax: ```js -let func = new Function(arg1, arg2, ..., body); +let func = new Function ([arg1, arg2, ...argN], functionBody); ``` -For historical reasons, arguments can also be given as a comma-separated list. +For historical reasons, arguments can also be given as a comma-separated list. -These three mean the same: +These three declarations mean the same: -```js +```js new Function('a', 'b', 'return a + b'); // basic syntax new Function('a,b', 'return a + b'); // comma-separated new Function('a , b', 'return a + b'); // comma-separated with spaces ``` -Functions created with `new Function`, have `[[Environment]]` referencing the global Lexical Environment, not the outer one. Hence, they cannot use outer variables. But that's actually good, because it saves us from errors. Passing parameters explicitly is a much better method architecturally and causes no problems with minifiers. +Functions created with `new Function`, have `[[Environment]]` referencing the global Lexical Environment, not the outer one. Hence, they cannot use outer variables. But that's actually good, because it insures us from errors. Passing parameters explicitly is a much better method architecturally and causes no problems with minifiers. diff --git a/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/solution.md b/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/solution.md index 13f01debf..b5b1da7a6 100644 --- a/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/solution.md +++ b/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/solution.md @@ -18,7 +18,7 @@ function printNumbers(from, to) { printNumbers(5, 10); ``` -Using recursive `setTimeout`: +Using nested `setTimeout`: ```js run @@ -38,5 +38,27 @@ function printNumbers(from, to) { printNumbers(5, 10); ``` -Note that in both solutions, there is an initial delay before the first output. Sometimes we need to add a line to make the first output immediately, that's easy to do. +Note that in both solutions, there is an initial delay before the first output. The function is called after `1000ms` the first time. +If we also want the function to run immediately, then we can add an additional call on a separate line, like this: + +```js run +function printNumbers(from, to) { + let current = from; + + function go() { + alert(current); + if (current == to) { + clearInterval(timerId); + } + current++; + } + +*!* + go(); +*/!* + let timerId = setInterval(go, 1000); +} + +printNumbers(5, 10); +``` diff --git a/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/task.md b/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/task.md index 87e723c67..84bb0c39c 100644 --- a/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/task.md +++ b/1-js/06-advanced-functions/08-settimeout-setinterval/1-output-numbers-100ms/task.md @@ -9,5 +9,4 @@ Write a function `printNumbers(from, to)` that outputs a number every second, st Make two variants of the solution. 1. Using `setInterval`. -2. Using recursive `setTimeout`. - +2. Using nested `setTimeout`. diff --git a/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/solution.md b/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/solution.md deleted file mode 100644 index 735a446f7..000000000 --- a/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/solution.md +++ /dev/null @@ -1,22 +0,0 @@ - - -```js run -let i = 0; - -let start = Date.now(); - -let timer = setInterval(count); - -function count() { - - for(let j = 0; j < 1000000; j++) { - i++; - } - - if (i == 1000000000) { - alert("Done in " + (Date.now() - start) + 'ms'); - clearInterval(timer); - } - -} -``` diff --git a/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/task.md b/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/task.md deleted file mode 100644 index c3455c2a1..000000000 --- a/1-js/06-advanced-functions/08-settimeout-setinterval/3-rewrite-settimeout/task.md +++ /dev/null @@ -1,32 +0,0 @@ -importance: 4 - ---- - -# Rewrite setTimeout with setInterval - -Here's the function that uses nested `setTimeout` to split a job into pieces. - -Rewrite it to `setInterval`: - -```js run -let i = 0; - -let start = Date.now(); - -function count() { - - if (i == 1000000000) { - alert("Done in " + (Date.now() - start) + 'ms'); - } else { - setTimeout(count); - } - - // a piece of heavy job - for(let j = 0; j < 1000000; j++) { - i++; - } - -} - -count(); -``` diff --git a/1-js/06-advanced-functions/08-settimeout-setinterval/article.md b/1-js/06-advanced-functions/08-settimeout-setinterval/article.md index a79ab0308..f96959988 100644 --- a/1-js/06-advanced-functions/08-settimeout-setinterval/article.md +++ b/1-js/06-advanced-functions/08-settimeout-setinterval/article.md @@ -4,12 +4,11 @@ We may decide to execute a function not right now, but at a certain time later. There are two methods for it: -- `setTimeout` allows to run a function once after the interval of time. -- `setInterval` allows to run a function regularly with the interval between the runs. +- `setTimeout` allows us to run a function once after the interval of time. +- `setInterval` allows us to run a function repeatedly, starting after the interval of time, then repeating continuously at that interval. These methods are not a part of JavaScript specification. But most environments have the internal scheduler and provide these methods. In particular, they are supported in all browsers and Node.js. - ## setTimeout The syntax: @@ -28,7 +27,7 @@ Usually, that's a function. For historical reasons, a string of code can be pass : The delay before run, in milliseconds (1000 ms = 1 second), by default 0. `arg1`, `arg2`... -: Arguments for the function (not supported in IE9-) +: Arguments for the function For instance, this code calls `sayHi()` after one second: @@ -62,7 +61,7 @@ So, this will also work: setTimeout("alert('Hello')", 1000); ``` -But using strings is not recommended, use functions instead of them, like this: +But using strings is not recommended, use arrow functions instead of them, like this: ```js run no-beautify setTimeout(() => alert('Hello'), 1000); @@ -75,7 +74,7 @@ Novice developers sometimes make a mistake by adding brackets `()` after the fun // wrong! setTimeout(sayHi(), 1000); ``` -That doesn't work, because `setTimeout` expects a reference to function. And here `sayHi()` runs the function, and the *result of its execution* is passed to `setTimeout`. In our case the result of `sayHi()` is `undefined` (the function returns nothing), so nothing is scheduled. +That doesn't work, because `setTimeout` expects a reference to a function. And here `sayHi()` runs the function, and the *result of its execution* is passed to `setTimeout`. In our case the result of `sayHi()` is `undefined` (the function returns nothing), so nothing is scheduled. ```` ### Canceling with clearTimeout @@ -103,7 +102,7 @@ As we can see from `alert` output, in a browser the timer identifier is a number Again, there is no universal specification for these methods, so that's fine. -For browsers, timers are described in the [timers section](https://www.w3.org/TR/html5/webappapis.html#timers) of HTML5 standard. +For browsers, timers are described in the [timers section](https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#timers) of HTML Living Standard. ## setInterval @@ -127,17 +126,17 @@ let timerId = setInterval(() => alert('tick'), 2000); setTimeout(() => { clearInterval(timerId); alert('stop'); }, 5000); ``` -```smart header="Modal windows freeze time in Chrome/Opera/Safari" -In browsers IE and Firefox the internal timer continues "ticking" while showing `alert/confirm/prompt`, but in Chrome, Opera and Safari the internal timer becomes "frozen". +```smart header="Time goes on while `alert` is shown" +In most browsers, including Chrome and Firefox the internal timer continues "ticking" while showing `alert/confirm/prompt`. -So if you run the code above and don't dismiss the `alert` window for some time, then in Firefox/IE next `alert` will be shown immediately as you do it (2 seconds passed from the previous invocation), and in Chrome/Opera/Safari -- after 2 more seconds (timer did not tick during the `alert`). +So if you run the code above and don't dismiss the `alert` window for some time, then the next `alert` will be shown immediately as you do it. The actual interval between alerts will be shorter than 2 seconds. ``` -## Recursive setTimeout +## Nested setTimeout There are two ways of running something regularly. -One is `setInterval`. The other one is a recursive `setTimeout`, like this: +One is `setInterval`. The other one is a nested `setTimeout`, like this: ```js /** instead of: @@ -154,7 +153,7 @@ let timerId = setTimeout(function tick() { The `setTimeout` above schedules the next call right at the end of the current one `(*)`. -The recursive `setTimeout` is a more flexible method than `setInterval`. This way the next call may be scheduled differently, depending on the results of the current one. +The nested `setTimeout` is a more flexible method than `setInterval`. This way the next call may be scheduled differently, depending on the results of the current one. For instance, we need to write a service that sends a request to the server every 5 seconds asking for data, but in case the server is overloaded, it should increase the interval to 10, 20, 40 seconds... @@ -176,30 +175,30 @@ let timerId = setTimeout(function request() { ``` -And if we regularly have CPU-hungry tasks, then we can measure the time taken by the execution and plan the next call sooner or later. +And if the functions that we're scheduling are CPU-hungry, then we can measure the time taken by the execution and plan the next call sooner or later. -**Recursive `setTimeout` guarantees a delay between the executions, `setInterval` -- does not.** +**Nested `setTimeout` allows to set the delay between the executions more precisely than `setInterval`.** Let's compare two code fragments. The first one uses `setInterval`: ```js let i = 1; setInterval(function() { - func(i); + func(i++); }, 100); ``` -The second one uses recursive `setTimeout`: +The second one uses nested `setTimeout`: ```js let i = 1; setTimeout(function run() { - func(i); + func(i++); setTimeout(run, 100); }, 100); ``` -For `setInterval` the internal scheduler will run `func(i)` every 100ms: +For `setInterval` the internal scheduler will run `func(i++)` every 100ms:  @@ -215,15 +214,15 @@ In this case the engine waits for `func` to complete, then checks the scheduler In the edge case, if the function always executes longer than `delay` ms, then the calls will happen without a pause at all. -And here is the picture for the recursive `setTimeout`: +And here is the picture for the nested `setTimeout`:  -**The recursive `setTimeout` guarantees the fixed delay (here 100ms).** +**The nested `setTimeout` guarantees the fixed delay (here 100ms).** That's because a new call is planned at the end of the previous one. -````smart header="Garbage collection" +````smart header="Garbage collection and setInterval/setTimeout callback" When a function is passed in `setInterval/setTimeout`, an internal reference is created to it and saved in the scheduler. It prevents the function from being garbage collected, even if there are no other references to it. ```js @@ -233,16 +232,16 @@ setTimeout(function() {...}, 100); For `setInterval` the function stays in memory until `clearInterval` is called. -There's a side-effect. A function references the outer lexical environment, so, while it lives, outer variables live too. They may take much more memory than the function itself. So when we don't need the scheduled function anymore, it's better to cancel it, even if it's very small. +There's a side effect. A function references the outer lexical environment, so, while it lives, outer variables live too. They may take much more memory than the function itself. So when we don't need the scheduled function anymore, it's better to cancel it, even if it's very small. ```` -## setTimeout(...,0) +## Zero delay setTimeout There's a special use case: `setTimeout(func, 0)`, or just `setTimeout(func)`. -This schedules the execution of `func` as soon as possible. But scheduler will invoke it only after the current code is complete. +This schedules the execution of `func` as soon as possible. But the scheduler will invoke it only after the currently executing script is complete. -So the function is scheduled to run "right after" the current code. In other words, *asynchronously*. +So the function is scheduled to run "right after" the current script. For instance, this outputs "Hello", then immediately "World": @@ -252,116 +251,14 @@ setTimeout(() => alert("World")); alert("Hello"); ``` -The first line "puts the call into calendar after 0ms". But the scheduler will only "check the calendar" after the current code is complete, so `"Hello"` is first, and `"World"` -- after it. - -### Splitting CPU-hungry tasks - -There's a trick to split CPU-hungry tasks using `setTimeout`. - -For instance, a syntax-highlighting script (used to colorize code examples on this page) is quite CPU-heavy. To highlight the code, it performs the analysis, creates many colored elements, adds them to the document -- for a big text that takes a lot. It may even cause the browser to "hang", which is unacceptable. - -So we can split the long text into pieces. First 100 lines, then plan another 100 lines using `setTimeout(..., 0)`, and so on. - -For clarity, let's take a simpler example for consideration. We have a function to count from `1` to `1000000000`. - -If you run it, the CPU will hang. For server-side JS that's clearly noticeable, and if you are running it in-browser, then try to click other buttons on the page -- you'll see that whole JavaScript actually is paused, no other actions work until it finishes. - -```js run -let i = 0; - -let start = Date.now(); - -function count() { - - // do a heavy job - for (let j = 0; j < 1e9; j++) { - i++; - } - - alert("Done in " + (Date.now() - start) + 'ms'); -} - -count(); -``` - -The browser may even show "the script takes too long" warning (but hopefully it won't, because the number is not very big). - -Let's split the job using the nested `setTimeout`: - -```js run -let i = 0; - -let start = Date.now(); - -function count() { - - // do a piece of the heavy job (*) - do { - i++; - } while (i % 1e6 != 0); - - if (i == 1e9) { - alert("Done in " + (Date.now() - start) + 'ms'); - } else { - setTimeout(count); // schedule the new call (**) - } - -} - -count(); -``` - -Now the browser UI is fully functional during the "counting" process. - -We do a part of the job `(*)`: - -1. First run: `i=1...1000000`. -2. Second run: `i=1000001..2000000`. -3. ...and so on, the `while` checks if `i` is evenly divided by `1000000`. - -Then the next call is scheduled in `(**)` if we're not done yet. - -Pauses between `count` executions provide just enough "breath" for the JavaScript engine to do something else, to react to other user actions. - -The notable thing is that both variants -- with and without splitting the job by `setTimeout` -- are comparable in speed. There's no much difference in the overall counting time. - -To make them closer, let's make an improvement. - -We'll move the scheduling in the beginning of the `count()`: - -```js run -let i = 0; - -let start = Date.now(); - -function count() { - - // move the scheduling at the beginning - if (i < 1e9 - 1e6) { - setTimeout(count); // schedule the new call - } - - do { - i++; - } while (i % 1e6 != 0); - - if (i == 1e9) { - alert("Done in " + (Date.now() - start) + 'ms'); - } - -} - -count(); -``` - -Now when we start to `count()` and know that we'll need to `count()` more, we schedule that immediately, before doing the job. +The first line "puts the call into calendar after 0ms". But the scheduler will only "check the calendar" after the current script is complete, so `"Hello"` is first, and `"World"` -- after it. -If you run it, it's easy to notice that it takes significantly less time. +There are also advanced browser-related use cases of zero-delay timeout, that we'll discuss in the chapter <info:event-loop>. -````smart header="Minimal delay of nested timers in-browser" -In the browser, there's a limitation of how often nested timers can run. The [HTML5 standard](https://www.w3.org/TR/html5/webappapis.html#timers) says: "after five nested timers, the interval is forced to be at least four milliseconds.". +````smart header="Zero delay is in fact not zero (in a browser)" +In the browser, there's a limitation of how often nested timers can run. The [HTML Living Standard](https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#timers) says: "after five nested timers, the interval is forced to be at least 4 milliseconds.". -Let's demonstrate what it means with the example below. The `setTimeout` call in it re-schedules itself after `0ms`. Each call remembers the real time from the previous one in the `times` array. What do the real delays look like? Let's see: +Let's demonstrate what it means with the example below. The `setTimeout` call in it re-schedules itself with zero delay. Each call remembers the real time from the previous one in the `times` array. What do the real delays look like? Let's see: ```js run let start = Date.now(); @@ -378,85 +275,28 @@ setTimeout(function run() { // 1,1,1,1,9,15,20,24,30,35,40,45,50,55,59,64,70,75,80,85,90,95,100 ``` -First timers run immediately (just as written in the spec), and then the delay comes into play and we see `9, 15, 20, 24...`. +First timers run immediately (just as written in the spec), and then we see `9, 15, 20, 24...`. The 4+ ms obligatory delay between invocations comes into play. + +The similar thing happens if we use `setInterval` instead of `setTimeout`: `setInterval(f)` runs `f` few times with zero-delay, and afterwards with 4+ ms delay. That limitation comes from ancient times and many scripts rely on it, so it exists for historical reasons. -For server-side JavaScript, that limitation does not exist, and there exist other ways to schedule an immediate asynchronous job, like [process.nextTick](https://nodejs.org/api/process.html) and [setImmediate](https://nodejs.org/api/timers.html) for Node.js. So the notion is browser-specific only. +For server-side JavaScript, that limitation does not exist, and there exist other ways to schedule an immediate asynchronous job, like [setImmediate](https://nodejs.org/api/timers.html#timers_setimmediate_callback_args) for Node.js. So this note is browser-specific. ```` -### Allowing the browser to render - -Another benefit for in-browser scripts is that they can show a progress bar or something to the user. That's because the browser usually does all "repainting" after the script is complete. - -So if we do a single huge function then even if it changes something, the changes are not reflected in the document till it finishes. - -Here's the demo: -```html run -<div id="progress"></div> - -<script> - let i = 0; - - function count() { - for (let j = 0; j < 1e6; j++) { - i++; - // put the current i into the <div> - // (we'll talk more about innerHTML in the specific chapter, should be obvious here) - progress.innerHTML = i; - } - } - - count(); -</script> -``` - -If you run it, the changes to `i` will show up after the whole count finishes. - -And if we use `setTimeout` to split it into pieces then changes are applied in-between the runs, so this looks better: - -```html run -<div id="progress"></div> - -<script> - let i = 0; - - function count() { - - // do a piece of the heavy job (*) - do { - i++; - progress.innerHTML = i; - } while (i % 1e3 != 0); - - if (i < 1e9) { - setTimeout(count); - } - - } - - count(); -</script> -``` - -Now the `<div>` shows increasing values of `i`. - ## Summary -- Methods `setInterval(func, delay, ...args)` and `setTimeout(func, delay, ...args)` allow to run the `func` regularly/once after `delay` milliseconds. -- To cancel the execution, we should call `clearInterval/clearTimeout` with the value returned by `setInterval/setTimeout`. -- Nested `setTimeout` calls is a more flexible alternative to `setInterval`. Also they can guarantee the minimal time *between* the executions. -- Zero-timeout scheduling `setTimeout(...,0)` is used to schedule the call "as soon as possible, but after the current code is complete". - -Some use cases of `setTimeout(...,0)`: -- To split CPU-hungry tasks into pieces, so that the script doesn't "hang" -- To let the browser do something else while the process is going on (paint the progress bar). +- Methods `setTimeout(func, delay, ...args)` and `setInterval(func, delay, ...args)` allow us to run the `func` once/regularly after `delay` milliseconds. +- To cancel the execution, we should call `clearTimeout/clearInterval` with the value returned by `setTimeout/setInterval`. +- Nested `setTimeout` calls are a more flexible alternative to `setInterval`, allowing us to set the time *between* executions more precisely. +- Zero delay scheduling with `setTimeout(func, 0)` (the same as `setTimeout(func)`) is used to schedule the call "as soon as possible, but after the current script is complete". +- The browser limits the minimal delay for five or more nested calls of `setTimeout` or for `setInterval` (after 5th call) to 4ms. That's for historical reasons. -Please note that all scheduling methods do not *guarantee* the exact delay. We should not rely on that in the scheduled code. +Please note that all scheduling methods do not *guarantee* the exact delay. For example, the in-browser timer may slow down for a lot of reasons: - The CPU is overloaded. - The browser tab is in the background mode. -- The laptop is on battery. +- The laptop is on battery saving mode. -All that may increase the minimal timer resolution (the minimal delay) to 300ms or even 1000ms depending on the browser and settings. +All that may increase the minimal timer resolution (the minimal delay) to 300ms or even 1000ms depending on the browser and OS-level performance settings. diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/_js.view/solution.js b/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/_js.view/solution.js index 9ef503703..d5a09efb3 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/_js.view/solution.js +++ b/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/_js.view/solution.js @@ -1,11 +1,12 @@ function spy(func) { function wrapper(...args) { + // using ...args instead of arguments to store "real" array in wrapper.calls wrapper.calls.push(args); - return func.apply(this, arguments); + return func.apply(this, args); } wrapper.calls = []; return wrapper; -} \ No newline at end of file +} diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/solution.md b/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/solution.md index 19a072014..0c8a211b4 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/solution.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/01-spy-decorator/solution.md @@ -1 +1 @@ -Here we can use `calls.push(args)` to store all arguments in the log and `f.apply(this, args)` to forward the call. +The wrapper returned by `spy(f)` should store all arguments and then use `f.apply` to forward the call. diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/solution.js b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/solution.js index 065a77d1f..661dd0cf4 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/solution.js +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/solution.js @@ -1,15 +1,7 @@ -function debounce(f, ms) { - - let isCooldown = false; - +function debounce(func, ms) { + let timeout; return function() { - if (isCooldown) return; - - f.apply(this, arguments); - - isCooldown = true; - - setTimeout(() => isCooldown = false, ms); + clearTimeout(timeout); + timeout = setTimeout(() => func.apply(this, arguments), ms); }; - -} \ No newline at end of file +} diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/test.js b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/test.js index 16dc171e1..750e649f8 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/test.js +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/_js.view/test.js @@ -1,41 +1,48 @@ -describe("debounce", function() { - before(function() { +describe('debounce', function () { + before(function () { this.clock = sinon.useFakeTimers(); }); - after(function() { + after(function () { this.clock.restore(); }); - it("calls the function at maximum once in ms milliseconds", function() { - let log = ''; + it('for one call - runs it after given ms', function () { + const f = sinon.spy(); + const debounced = debounce(f, 1000); - function f(a) { - log += a; - } + debounced('test'); + assert(f.notCalled, 'not called immediately'); + this.clock.tick(1000); + assert(f.calledOnceWith('test'), 'called after 1000ms'); + }); - f = debounce(f, 1000); + it('for 3 calls - runs the last one after given ms', function () { + const f = sinon.spy(); + const debounced = debounce(f, 1000); - f(1); // runs at once - f(2); // ignored + debounced('a'); + setTimeout(() => debounced('b'), 200); // ignored (too early) + setTimeout(() => debounced('c'), 500); // runs (1000 ms passed) + this.clock.tick(1000); - setTimeout(() => f(3), 100); // ignored (too early) - setTimeout(() => f(4), 1100); // runs (1000 ms passed) - setTimeout(() => f(5), 1500); // ignored (less than 1000 ms from the last run) + assert(f.notCalled, 'not called after 1000ms'); - this.clock.tick(5000); - assert.equal(log, "14"); + this.clock.tick(500); + + assert(f.calledOnceWith('c'), 'called after 1500ms'); }); - it("keeps the context of the call", function() { + it('keeps the context of the call', function () { let obj = { f() { assert.equal(this, obj); - } + }, }; obj.f = debounce(obj.f, 1000); - obj.f("test"); + obj.f('test'); + this.clock.tick(5000); }); - -}); \ No newline at end of file + +}); diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.svg b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.svg new file mode 100644 index 000000000..e624ce020 --- /dev/null +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="500" height="183" viewBox="0 0 500 183"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="combined" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="debounce.svg"><text id="200ms" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="117" y="142">200ms</tspan></text><text id="1500ms" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="321" y="142">1500ms</tspan></text><text id="1000ms" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="242" y="80">1000ms</tspan></text><path id="Shape" fill="#C06334" stroke="#C06334" stroke-linecap="square" stroke-width="3" d="M343.378 28.32L350.92 12 331 33.024l9.674 2.932L334.797 52 353 31.253l-9.622-2.932zm4.369-11.617l-5.669 12.226 9.206 2.767-13.887 15.989 4.525-12.338-9.206-2.766 15.031-15.878z"/><text id="0" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="142">0</tspan></text><text id="c" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="333" y="72">c</tspan></text><text id="f(a)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="81" y="164">f(a)</tspan></text><text id="f(b)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="121" y="164">f(b)</tspan></text><text id="f(c)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="181" y="164">f(c)</tspan></text><text id="500ms" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="177" y="142">500ms</tspan></text><text id="time" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="427" y="125">time</tspan></text><text id="calls:" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7" y="164">calls:</tspan></text><path id="Line" fill="#DBAF88" fill-rule="nonzero" d="M432.369 94.388l.871.49 12 6.75 1.55.872-1.55.872-12 6.75-.871.49-.98-1.743.87-.49 8.673-4.879H48v-2h392.932l-8.672-4.878-.872-.49.98-1.744z"/><text id="after-1000ms" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="358" y="36"> after 1000ms</tspan></text><path id="Line-Copy" stroke="#DBAF88" stroke-linecap="square" stroke-width="2" d="M137.5 125V80M97.5 125V80M197.5 125V80"/><path id="Line-2" stroke="#C06334" stroke-linecap="square" stroke-width="3" d="M337.5 125V80"/><path id="Line-25" fill="#DBAF88" fill-rule="nonzero" d="M212.631 77.888l.98 1.743-.87.49L204.066 85H206v2h-1.933l8.673 4.878.872.49-.98 1.744-.872-.49-12-6.75-1.55-.872 1.55-.872 12-6.75.871-.49zm108.738 0l.871.49 12 6.75 1.55.872-1.55.872-12 6.75-.871.49-.98-1.743.87-.49L329.933 87H329v-2h.932l-8.672-4.878-.872-.49.98-1.744zM214 85v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5zm8 0v2h-5v-2h5z"/></g></g></svg> \ No newline at end of file diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.view/index.html b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.view/index.html new file mode 100644 index 000000000..e3b4d5842 --- /dev/null +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/debounce.view/index.html @@ -0,0 +1,24 @@ +<!doctype html> +<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.11/lodash.js"></script> + +Function <code>handler</code> is called on this input: +<br> +<input id="input1" placeholder="type here"> + +<p> + +Debounced function <code>debounce(handler, 1000)</code> is called on this input: +<br> +<input id="input2" placeholder="type here"> + +<p> +<button id="result">The <code>handler</code> puts the result here</button> + +<script> + function handler(event) { + result.innerHTML = event.target.value; + } + + input1.oninput = handler; + input2.oninput = _.debounce(handler, 1000); +</script> \ No newline at end of file diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/solution.md b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/solution.md index 4f5867ded..83e75f315 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/solution.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/solution.md @@ -1,28 +1,13 @@ ```js demo -function debounce(f, ms) { - - let isCooldown = false; - +function debounce(func, ms) { + let timeout; return function() { - if (isCooldown) return; - - f.apply(this, arguments); - - isCooldown = true; - - setTimeout(() => isCooldown = false, ms); + clearTimeout(timeout); + timeout = setTimeout(() => func.apply(this, arguments), ms); }; - } -``` - -A call to `debounce` returns a wrapper. There may be two states: -- `isCooldown = false` -- ready to run. -- `isCooldown = true` -- waiting for the timeout. - -In the first call `isCooldown` is falsy, so the call proceeds, and the state changes to `true`. +``` -While `isCooldown` is true, all other calls are ignored. +A call to `debounce` returns a wrapper. When called, it schedules the original function call after given `ms` and cancels the previous such timeout. -Then `setTimeout` reverts it to `false` after the given delay. diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/task.md b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/task.md index 466c6bc3f..5b0fcc5f8 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/task.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/03-debounce/task.md @@ -4,21 +4,48 @@ importance: 5 # Debounce decorator -The result of `debounce(f, ms)` decorator should be a wrapper that passes the call to `f` at maximum once per `ms` milliseconds. +The result of `debounce(f, ms)` decorator is a wrapper that suspends calls to `f` until there's `ms` milliseconds of inactivity (no calls, "cooldown period"), then invokes `f` once with the latest arguments. -In other words, when we call a "debounced" function, it guarantees that all other future in the closest `ms` milliseconds will be ignored. +In other words, `debounce` is like a secretary that accepts "phone calls", and waits until there's `ms` milliseconds of being quiet. And only then it transfers the latest call information to "the boss" (calls the actual `f`). -For instance: +For instance, we had a function `f` and replaced it with `f = debounce(f, 1000)`. -```js no-beautify -let f = debounce(alert, 1000); +Then if the wrapped function is called at 0ms, 200ms and 500ms, and then there are no calls, then the actual `f` will be only called once, at 1500ms. That is: after the cooldown period of 1000ms from the last call. -f(1); // runs immediately -f(2); // ignored + -setTimeout( () => f(3), 100); // ignored ( only 100 ms passed ) -setTimeout( () => f(4), 1100); // runs -setTimeout( () => f(5), 1500); // ignored (less than 1000 ms from the last run) +...And it will get the arguments of the very last call, other calls are ignored. + +Here's the code for it (uses the debounce decorator from the [Lodash library](https://lodash.com/docs/4.17.15#debounce)): + +```js +let f = _.debounce(alert, 1000); + +f("a"); +setTimeout( () => f("b"), 200); +setTimeout( () => f("c"), 500); +// debounced function waits 1000ms after the last call and then runs: alert("c") +``` + +Now a practical example. Let's say, the user types something, and we'd like to send a request to the server when the input is finished. + +There's no point in sending the request for every character typed. Instead we'd like to wait, and then process the whole result. + +In a web-browser, we can setup an event handler -- a function that's called on every change of an input field. Normally, an event handler is called very often, for every typed key. But if we `debounce` it by 1000ms, then it will be only called once, after 1000ms after the last input. + +```online + +In this live example, the handler puts the result into a box below, try it: + +[iframe border=1 src="debounce" height=200] + +See? The second input calls the debounced function, so its content is processed after 1000ms from the last input. ``` -In practice `debounce` is useful for functions that retrieve/update something when we know that nothing new can be done in such a short period of time, so it's better not to waste resources. \ No newline at end of file +So, `debounce` is a great way to process a sequence of events: be it a sequence of key presses, mouse movements or something else. + +It waits the given time after the last call, and then runs its function, that can process the result. + +The task is to implement `debounce` decorator. + +Hint: that's just a few lines if you think about it :) diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/_js.view/test.js b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/_js.view/test.js index 5339c8d11..e671438f6 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/_js.view/test.js +++ b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/_js.view/test.js @@ -7,8 +7,8 @@ describe("throttle(f, 1000)", function() { } before(function() { - f1000 = throttle(f, 1000); this.clock = sinon.useFakeTimers(); + f1000 = throttle(f, 1000); }); it("the first call runs now", function() { @@ -44,4 +44,20 @@ describe("throttle(f, 1000)", function() { this.clock.restore(); }); -}); \ No newline at end of file +}); + +describe('throttle', () => { + + it('runs a forwarded call once', done => { + let log = ''; + const f = str => log += str; + const f10 = throttle(f, 10); + f10('once'); + + setTimeout(() => { + assert.equal(log, 'once'); + done(); + }, 20); + }); + +}); diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/solution.md b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/solution.md index c844016d3..6950664be 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/solution.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/solution.md @@ -12,11 +12,10 @@ function throttle(func, ms) { savedThis = this; return; } + isThrottled = true; func.apply(this, arguments); // (1) - isThrottled = true; - setTimeout(function() { isThrottled = false; // (3) if (savedArgs) { @@ -33,7 +32,7 @@ function throttle(func, ms) { A call to `throttle(func, ms)` returns `wrapper`. 1. During the first call, the `wrapper` just runs `func` and sets the cooldown state (`isThrottled = true`). -2. In this state all calls memorized in `savedArgs/savedThis`. Please note that both the context and the arguments are equally important and should be memorized. We need them simultaneously to reproduce the call. -3. ...Then after `ms` milliseconds pass, `setTimeout` triggers. The cooldown state is removed (`isThrottled = false`). And if we had ignored calls, then `wrapper` is executed with last memorized arguments and context. +2. In this state all calls are memorized in `savedArgs/savedThis`. Please note that both the context and the arguments are equally important and should be memorized. We need them simultaneously to reproduce the call. +3. After `ms` milliseconds pass, `setTimeout` triggers. The cooldown state is removed (`isThrottled = false`) and, if we had ignored calls, `wrapper` is executed with the last memorized arguments and context. The 3rd step runs not `func`, but `wrapper`, because we not only need to execute `func`, but once again enter the cooldown state and setup the timeout to reset it. diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/task.md b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/task.md index 8dd77368d..cbd473196 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/task.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/04-throttle/task.md @@ -4,35 +4,40 @@ importance: 5 # Throttle decorator -Create a "throttling" decorator `throttle(f, ms)` -- that returns a wrapper, passing the call to `f` at maximum once per `ms` milliseconds. Those calls that fall into the "cooldown" period, are ignored. +Create a "throttling" decorator `throttle(f, ms)` -- that returns a wrapper. -**The difference with `debounce` -- if an ignored call is the last during the cooldown, then it executes at the end of the delay.** +When it's called multiple times, it passes the call to `f` at maximum once per `ms` milliseconds. + +Compared to the debounce decorator, the behavior is completely different: +- `debounce` runs the function once after the "cooldown" period. Good for processing the final result. +- `throttle` runs it not more often than given `ms` time. Good for regular updates that shouldn't be very often. + +In other words, `throttle` is like a secretary that accepts phone calls, but bothers the boss (calls the actual `f`) not more often than once per `ms` milliseconds. Let's check the real-life application to better understand that requirement and to see where it comes from. **For instance, we want to track mouse movements.** -In browser we can setup a function to run at every mouse movement and get the pointer location as it moves. During an active mouse usage, this function usually runs very frequently, can be something like 100 times per second (every 10 ms). - -**The tracking function should update some information on the web-page.** +In a browser we can setup a function to run at every mouse movement and get the pointer location as it moves. During an active mouse usage, this function usually runs very frequently, can be something like 100 times per second (every 10 ms). +**We'd like to update some information on the web-page when the pointer moves.** -Updating function `update()` is too heavy to do it on every micro-movement. There is also no sense in making it more often than once per 100ms. +...But updating function `update()` is too heavy to do it on every micro-movement. There is also no sense in updating more often than once per 100ms. -So we'll wrap it into the decorator: use `throttle(update, 100)` as the function to run on each mouse move instead of the original `update()`. The decorator will be called often, but `update()` will be called at maximum once per 100ms. +So we'll wrap it into the decorator: use `throttle(update, 100)` as the function to run on each mouse move instead of the original `update()`. The decorator will be called often, but forward the call to `update()` at maximum once per 100ms. Visually, it will look like this: -1. For the first mouse movement the decorated variant passes the call to `update`. That's important, the user sees our reaction to their move immediately. +1. For the first mouse movement the decorated variant immediately passes the call to `update`. That's important, the user sees our reaction to their move immediately. 2. Then as the mouse moves on, until `100ms` nothing happens. The decorated variant ignores calls. -3. At the end of `100ms` -- one more `update` happens with the last coordinates. -4. Then, finally, the mouse stops somewhere. The decorated variant waits until `100ms` expire and then runs `update` with last coordinates. So, perhaps the most important, the final mouse coordinates are processed. +3. At the end of `100ms` -- one more `update` happens with the last coordinates. +4. Then, finally, the mouse stops somewhere. The decorated variant waits until `100ms` expire and then runs `update` with last coordinates. So, quite important, the final mouse coordinates are processed. A code example: ```js function f(a) { - console.log(a) -}; + console.log(a); +} // f1000 passes calls to f at maximum once per 1000 ms let f1000 = throttle(f, 1000); diff --git a/1-js/06-advanced-functions/09-call-apply-decorators/article.md b/1-js/06-advanced-functions/09-call-apply-decorators/article.md index f2182b6fd..c5d785493 100644 --- a/1-js/06-advanced-functions/09-call-apply-decorators/article.md +++ b/1-js/06-advanced-functions/09-call-apply-decorators/article.md @@ -6,9 +6,9 @@ JavaScript gives exceptional flexibility when dealing with functions. They can b Let's say we have a function `slow(x)` which is CPU-heavy, but its results are stable. In other words, for the same `x` it always returns the same result. -If the function is called often, we may want to cache (remember) the results for different `x` to avoid spending extra-time on recalculations. +If the function is called often, we may want to cache (remember) the results to avoid spending extra-time on recalculations. -But instead of adding that functionality into `slow()` we'll create a wrapper. As we'll see, there are many benefits of doing so. +But instead of adding that functionality into `slow()` we'll create a wrapper function, that adds caching. As we'll see, there are many benefits of doing so. Here's the code, and explanations follow: @@ -23,24 +23,24 @@ function cachingDecorator(func) { let cache = new Map(); return function(x) { - if (cache.has(x)) { // if the result is in the map - return cache.get(x); // return it + if (cache.has(x)) { // if there's such key in cache + return cache.get(x); // read the result from it } - let result = func(x); // otherwise call func + let result = func(x); // otherwise call func - cache.set(x, result); // and cache (remember) the result + cache.set(x, result); // and cache (remember) the result return result; }; } slow = cachingDecorator(slow); -alert( slow(1) ); // slow(1) is cached -alert( "Again: " + slow(1) ); // the same +alert( slow(1) ); // slow(1) is cached and the result returned +alert( "Again: " + slow(1) ); // slow(1) result returned from cache -alert( slow(2) ); // slow(2) is cached -alert( "Again: " + slow(2) ); // the same as the previous line +alert( slow(2) ); // slow(2) is cached and the result returned +alert( "Again: " + slow(2) ); // slow(2) result returned from cache ``` In the code above `cachingDecorator` is a *decorator*: a special function that takes another function and alters its behavior. @@ -49,21 +49,18 @@ The idea is that we can call `cachingDecorator` for any function, and it will re By separating caching from the main function code we also keep the main code simpler. -Now let's get into details of how it works. - The result of `cachingDecorator(func)` is a "wrapper": `function(x)` that "wraps" the call of `func(x)` into caching logic:  -As we can see, the wrapper returns the result of `func(x)` "as is". From an outside code, the wrapped `slow` function still does the same. It just got a caching aspect added to its behavior. +From an outside code, the wrapped `slow` function still does the same. It just got a caching aspect added to its behavior. To summarize, there are several benefits of using a separate `cachingDecorator` instead of altering the code of `slow` itself: - The `cachingDecorator` is reusable. We can apply it to another function. -- The caching logic is separate, it did not increase the complexity of `slow` itself (if there were any). +- The caching logic is separate, it did not increase the complexity of `slow` itself (if there was any). - We can combine multiple decorators if needed (other decorators will follow). - ## Using "func.call" for the context The caching decorator mentioned above is not suited to work with object methods. @@ -78,7 +75,7 @@ let worker = { }, slow(x) { - // actually, there can be a scary CPU-heavy task here + // scary CPU-heavy task here alert("Called with " + x); return x * this.someMethod(); // (*) } @@ -152,8 +149,8 @@ let user = { name: "John" }; let admin = { name: "Admin" }; // use call to pass different objects as "this" -sayHi.call( user ); // this = John -sayHi.call( admin ); // this = Admin +sayHi.call( user ); // John +sayHi.call( admin ); // Admin ``` And here we use `call` to call `say` with the given context and phrase: @@ -170,10 +167,8 @@ let user = { name: "John" }; say.call( user, "Hello" ); // John: Hello ``` - In our case, we can use `call` in the wrapper to pass the context to the original function: - ```js run let worker = { someMethod() { @@ -214,7 +209,7 @@ To make it all clear, let's see more deeply how `this` is passed along: 2. So when `worker.slow(2)` is executed, the wrapper gets `2` as an argument and `this=worker` (it's the object before dot). 3. Inside the wrapper, assuming the result is not yet cached, `func.call(this, x)` passes the current `this` (`=worker`) and the current argument (`=2`) to the original method. -## Going multi-argument with "func.apply" +## Going multi-argument Now let's make `cachingDecorator` even more universal. Till now it was working only with single-argument functions. @@ -231,9 +226,7 @@ let worker = { worker.slow = cachingDecorator(worker.slow); ``` -We have two tasks to solve here. - -First is how to use both arguments `min` and `max` for the key in `cache` map. Previously, for a single argument `x` we could just `cache.set(x, result)` to save the result and `cache.get(x)` to retrieve it. But now we need to remember the result for a *combination of arguments* `(min,max)`. The native `Map` takes single value only as the key. +Previously, for a single argument `x` we could just `cache.set(x, result)` to save the result and `cache.get(x)` to retrieve it. But now we need to remember the result for a *combination of arguments* `(min,max)`. The native `Map` takes single value only as the key. There are many solutions possible: @@ -241,85 +234,11 @@ There are many solutions possible: 2. Use nested maps: `cache.set(min)` will be a `Map` that stores the pair `(max, result)`. So we can get `result` as `cache.get(min).get(max)`. 3. Join two values into one. In our particular case we can just use a string `"min,max"` as the `Map` key. For flexibility, we can allow to provide a *hashing function* for the decorator, that knows how to make one value from many. - For many practical applications, the 3rd variant is good enough, so we'll stick to it. -The second task to solve is how to pass many arguments to `func`. Currently, the wrapper `function(x)` assumes a single argument, and `func.call(this, x)` passes it. - -Here we can use another built-in method [func.apply](mdn:js/Function/apply). - -The syntax is: - -```js -func.apply(context, args) -``` - -It runs the `func` setting `this=context` and using an array-like object `args` as the list of arguments. - - -For instance, these two calls are almost the same: - -```js -func(1, 2, 3); -func.apply(context, [1, 2, 3]) -``` - -Both run `func` giving it arguments `1,2,3`. But `apply` also sets `this=context`. - -For instance, here `say` is called with `this=user` and `messageData` as a list of arguments: - -```js run -function say(time, phrase) { - alert(`[${time}] ${this.name}: ${phrase}`); -} - -let user = { name: "John" }; - -let messageData = ['10:00', 'Hello']; // become time and phrase - -*!* -// user becomes this, messageData is passed as a list of arguments (time, phrase) -say.apply(user, messageData); // [10:00] John: Hello (this=user) -*/!* -``` - -The only syntax difference between `call` and `apply` is that `call` expects a list of arguments, while `apply` takes an array-like object with them. - -We already know the spread operator `...` from the chapter <info:rest-parameters-spread-operator> that can pass an array (or any iterable) as a list of arguments. So if we use it with `call`, we can achieve almost the same as `apply`. - -These two calls are almost equivalent: - -```js -let args = [1, 2, 3]; - -*!* -func.call(context, ...args); // pass an array as list with spread operator -func.apply(context, args); // is same as using apply -*/!* -``` - -If we look more closely, there's a minor difference between such uses of `call` and `apply`. - -- The spread operator `...` allows to pass *iterable* `args` as the list to `call`. -- The `apply` accepts only *array-like* `args`. - -So, these calls complement each other. Where we expect an iterable, `call` works, where we expect an array-like, `apply` works. - -And if `args` is both iterable and array-like, like a real array, then we technically could use any of them, but `apply` will probably be faster, because it's a single operation. Most JavaScript engines internally optimize it better than a pair `call + spread`. - -One of the most important uses of `apply` is passing the call to another function, like this: - -```js -let wrapper = function() { - return anotherFunction.apply(this, arguments); -}; -``` - -That's called *call forwarding*. The `wrapper` passes everything it gets: the context `this` and arguments to `anotherFunction` and returns back its result. +Also we need to pass not just `x`, but all arguments in `func.call`. Let's recall that in a `function()` we can get a pseudo-array of its arguments as `arguments`, so `func.call(this, x)` should be replaced with `func.call(this, ...arguments)`. -When an external code calls such `wrapper`, it is indistinguishable from the call of the original function. - -Now let's bake it all into the more powerful `cachingDecorator`: +Here's a more powerful `cachingDecorator`: ```js run let worker = { @@ -340,7 +259,7 @@ function cachingDecorator(func, hash) { } *!* - let result = func.apply(this, arguments); // (**) + let result = func.call(this, ...arguments); // (**) */!* cache.set(key, result); @@ -358,13 +277,54 @@ alert( worker.slow(3, 5) ); // works alert( "Again " + worker.slow(3, 5) ); // same (cached) ``` -Now the wrapper operates with any number of arguments. +Now it works with any number of arguments (though the hash function would also need to be adjusted to allow any number of arguments. An interesting way to handle this will be covered below). There are two changes: - In the line `(*)` it calls `hash` to create a single key from `arguments`. Here we use a simple "joining" function that turns arguments `(3, 5)` into the key `"3,5"`. More complex cases may require other hashing functions. -- Then `(**)` uses `func.apply` to pass both the context and all arguments the wrapper got (no matter how many) to the original function. +- Then `(**)` uses `func.call(this, ...arguments)` to pass both the context and all arguments the wrapper got (not just the first one) to the original function. + +## func.apply + +Instead of `func.call(this, ...arguments)` we could use `func.apply(this, arguments)`. + +The syntax of built-in method [func.apply](mdn:js/Function/apply) is: + +```js +func.apply(context, args) +``` + +It runs the `func` setting `this=context` and using an array-like object `args` as the list of arguments. + +The only syntax difference between `call` and `apply` is that `call` expects a list of arguments, while `apply` takes an array-like object with them. + +So these two calls are almost equivalent: + +```js +func.call(context, ...args); +func.apply(context, args); +``` + +They perform the same call of `func` with given context and arguments. + +There's only a subtle difference regarding `args`: + +- The spread syntax `...` allows to pass *iterable* `args` as the list to `call`. +- The `apply` accepts only *array-like* `args`. +...And for objects that are both iterable and array-like, such as a real array, we can use any of them, but `apply` will probably be faster, because most JavaScript engines internally optimize it better. + +Passing all arguments along with the context to another function is called *call forwarding*. + +That's the simplest form of it: + +```js +let wrapper = function() { + return func.apply(this, arguments); +}; +``` + +When an external code calls such `wrapper`, it is indistinguishable from the call of the original function `func`. ## Borrowing a method [#method-borrowing] @@ -386,7 +346,7 @@ function hash(args) { } ``` -...Unfortunately, that won't work. Because we are calling `hash(arguments)` and `arguments` object is both iterable and array-like, but not a real array. +...Unfortunately, that won't work. Because we are calling `hash(arguments)`, and `arguments` object is both iterable and array-like, but not a real array. So calling `join` on it would fail, as we can see below: @@ -414,7 +374,7 @@ hash(1, 2); The trick is called *method borrowing*. -We take (borrow) a join method from a regular array `[].join`. And use `[].join.call` to run it in the context of `arguments`. +We take (borrow) a join method from a regular array (`[].join`) and use `[].join.call` to run it in the context of `arguments`. Why does it work? @@ -432,12 +392,20 @@ Taken from the specification almost "as-is": So, technically it takes `this` and joins `this[0]`, `this[1]` ...etc together. It's intentionally written in a way that allows any array-like `this` (not a coincidence, many methods follow this practice). That's why it also works with `this=arguments`. +## Decorators and function properties + +It is generally safe to replace a function or a method with a decorated one, except for one little thing. If the original function had properties on it, like `func.calledCount` or whatever, then the decorated one will not provide them. Because that is a wrapper. So one needs to be careful if one uses them. + +E.g. in the example above if `slow` function had any properties on it, then `cachingDecorator(slow)` is a wrapper without them. + +Some decorators may provide their own properties. E.g. a decorator may count how many times a function was invoked and how much time it took, and expose this information via wrapper properties. + +There exists a way to create decorators that keep access to function properties, but this requires using a special `Proxy` object to wrap a function. We'll discuss it later in the article <info:proxy#proxy-apply>. + ## Summary *Decorator* is a wrapper around a function that alters its behavior. The main job is still carried out by the function. -It is generally safe to replace a function or a method with a decorated one, except for one little thing. If the original function had properties on it, like `func.calledCount` or whatever, then the decorated one will not provide them. Because that is a wrapper. So one needs to be careful if one uses them. Some decorators provide their own properties. - Decorators can be seen as "features" or "aspects" that can be added to a function. We can add one or add many. And all this without changing its code! To implement `cachingDecorator`, we studied methods: @@ -450,10 +418,9 @@ The generic *call forwarding* is usually done with `apply`: ```js let wrapper = function() { return original.apply(this, arguments); -} +}; ``` -We also saw an example of *method borrowing* when we take a method from an object and `call` it in the context of another object. It is quite common to take array methods and apply them to arguments. The alternative is to use rest parameters object that is a real array. - +We also saw an example of *method borrowing* when we take a method from an object and `call` it in the context of another object. It is quite common to take array methods and apply them to `arguments`. The alternative is to use rest parameters object that is a real array. There are many decorators there in the wild. Check how well you got them by solving the tasks of this chapter. diff --git a/1-js/06-advanced-functions/10-bind/4-function-property-after-bind/task.md b/1-js/06-advanced-functions/10-bind/4-function-property-after-bind/task.md index 8cd18ec56..d6cfb44bf 100644 --- a/1-js/06-advanced-functions/10-bind/4-function-property-after-bind/task.md +++ b/1-js/06-advanced-functions/10-bind/4-function-property-after-bind/task.md @@ -4,7 +4,7 @@ importance: 5 # Function property after bind -There's a value in the property of a function. Will it change after `bind`? Why, elaborate? +There's a value in the property of a function. Will it change after `bind`? Why, or why not? ```js run function sayHi() { diff --git a/1-js/06-advanced-functions/10-bind/5-question-use-bind/solution.md b/1-js/06-advanced-functions/10-bind/5-question-use-bind/solution.md index 0cb673b12..4a381c0b4 100644 --- a/1-js/06-advanced-functions/10-bind/5-question-use-bind/solution.md +++ b/1-js/06-advanced-functions/10-bind/5-question-use-bind/solution.md @@ -1,5 +1,5 @@ -The error occurs because `ask` gets functions `loginOk/loginFail` without the object. +The error occurs because `askPassword` gets functions `loginOk/loginFail` without the object. When it calls them, they naturally assume `this=undefined`. @@ -38,6 +38,6 @@ An alternative solution could be: askPassword(() => user.loginOk(), () => user.loginFail()); ``` -Usually that also works, but may fail in more complex situations where `user` has a chance of being overwritten between the moments of asking and running `() => user.loginOk()`. - +Usually that also works and looks good. +It's a bit less reliable though in more complex situations where `user` variable might change *after* `askPassword` is called, but *before* the visitor answers and calls `() => user.loginOk()`. diff --git a/1-js/06-advanced-functions/10-bind/5-question-use-bind/task.md b/1-js/06-advanced-functions/10-bind/5-question-use-bind/task.md index eb19e6644..fe6a9b4eb 100644 --- a/1-js/06-advanced-functions/10-bind/5-question-use-bind/task.md +++ b/1-js/06-advanced-functions/10-bind/5-question-use-bind/task.md @@ -2,7 +2,7 @@ importance: 5 --- -# Ask losing this +# Fix a function that loses "this" The call to `askPassword()` in the code below should check the password and then call `user.loginOk/loginFail` depending on the answer. @@ -34,5 +34,3 @@ let user = { askPassword(user.loginOk, user.loginFail); */!* ``` - - diff --git a/1-js/06-advanced-functions/11-currying-partials/1-ask-currying/solution.md b/1-js/06-advanced-functions/10-bind/6-ask-partial/solution.md similarity index 100% rename from 1-js/06-advanced-functions/11-currying-partials/1-ask-currying/solution.md rename to 1-js/06-advanced-functions/10-bind/6-ask-partial/solution.md diff --git a/1-js/06-advanced-functions/11-currying-partials/1-ask-currying/task.md b/1-js/06-advanced-functions/10-bind/6-ask-partial/task.md similarity index 82% rename from 1-js/06-advanced-functions/11-currying-partials/1-ask-currying/task.md rename to 1-js/06-advanced-functions/10-bind/6-ask-partial/task.md index f8b83d7a2..c90851c2b 100644 --- a/1-js/06-advanced-functions/11-currying-partials/1-ask-currying/task.md +++ b/1-js/06-advanced-functions/10-bind/6-ask-partial/task.md @@ -8,7 +8,7 @@ The task is a little more complex variant of <info:task/question-use-bind>. The `user` object was modified. Now instead of two functions `loginOk/loginFail`, it has a single function `user.login(true/false)`. -What to pass `askPassword` in the code below, so that it calls `user.login(true)` as `ok` and `user.login(false)` as `fail`? +What should we pass `askPassword` in the code below, so that it calls `user.login(true)` as `ok` and `user.login(false)` as `fail`? ```js function askPassword(ok, fail) { diff --git a/1-js/06-advanced-functions/10-bind/article.md b/1-js/06-advanced-functions/10-bind/article.md index 06e2000ff..7a6e47b90 100644 --- a/1-js/06-advanced-functions/10-bind/article.md +++ b/1-js/06-advanced-functions/10-bind/article.md @@ -5,13 +5,13 @@ libs: # Function binding -When using `setTimeout` with object methods or passing object methods along, there's a known problem: "losing `this`". +When passing object methods as callbacks, for instance to `setTimeout`, there's a known problem: "losing `this`". -Suddenly, `this` just stops working right. The situation is typical for novice developers, but happens with experienced ones as well. +In this chapter we'll see the ways to fix it. ## Losing "this" -We already know that in JavaScript it's easy to lose `this`. Once a method is passed somewhere separately from the object -- `this` is lost. +We've already seen examples of losing `this`. Once a method is passed somewhere separately from the object -- `this` is lost. Here's how it may happen with `setTimeout`: @@ -37,7 +37,7 @@ let f = user.sayHi; setTimeout(f, 1000); // lost user context ``` -The method `setTimeout` in-browser is a little special: it sets `this=window` for the function call (for Node.js, `this` becomes the timer object, but doesn't really matter here). So for `this.firstName` it tries to get `window.firstName`, which does not exist. In other similar cases as we'll see, usually `this` just becomes `undefined`. +The method `setTimeout` in-browser is a little special: it sets `this=window` for the function call (for Node.js, `this` becomes the timer object, but doesn't really matter here). So for `this.firstName` it tries to get `window.firstName`, which does not exist. In other similar cases, usually `this` just becomes `undefined`. The task is quite typical -- we want to pass an object method somewhere else (here -- to the scheduler) where it will be called. How to make sure that it will be called in the right context? @@ -83,10 +83,12 @@ let user = { setTimeout(() => user.sayHi(), 1000); -// ...within 1 second -user = { sayHi() { alert("Another user in setTimeout!"); } }; +// ...the value of user changes within 1 second +user = { + sayHi() { alert("Another user in setTimeout!"); } +}; -// Another user in setTimeout?!? +// Another user in setTimeout! ``` The next solution guarantees that such thing won't happen. @@ -98,7 +100,7 @@ Functions provide a built-in method [bind](mdn:js/Function/bind) that allows to The basic syntax is: ```js -// more complex syntax will be little later +// more complex syntax will come a little later let boundFunc = func.bind(context); ``` @@ -123,7 +125,7 @@ funcUser(); // John */!* ``` -Here `func.bind(user)` as a "bound variant" of `func`, with fixed `this=user`. +Here `func.bind(user)` is a "bound variant" of `func`, with fixed `this=user`. All arguments are passed to the original `func` "as is", for instance: @@ -159,9 +161,16 @@ let user = { let sayHi = user.sayHi.bind(user); // (*) */!* +// can run it without an object sayHi(); // Hello, John! setTimeout(sayHi, 1000); // Hello, John! + +// even if the value of user changes within 1 second +// sayHi uses the pre-bound value which is reference to the old user object +user = { + sayHi() { alert("Another user in setTimeout!"); } +}; ``` In the line `(*)` we take the method `user.sayHi` and bind it to `user`. The `sayHi` is a "bound" function, that can be called alone or passed to `setTimeout` -- doesn't matter, the context will be right. @@ -178,8 +187,8 @@ let user = { let say = user.say.bind(user); -say("Hello"); // Hello, John ("Hello" argument is passed to say) -say("Bye"); // Bye, John ("Bye" is passed to say) +say("Hello"); // Hello, John! ("Hello" argument is passed to say) +say("Bye"); // Bye, John! ("Bye" is passed to say) ``` ````smart header="Convenience method: `bindAll`" @@ -193,11 +202,127 @@ for (let key in user) { } ``` -JavaScript libraries also provide functions for convenient mass binding , e.g. [_.bindAll(obj)](http://lodash.com/docs#bindAll) in lodash. +JavaScript libraries also provide functions for convenient mass binding , e.g. [_.bindAll(object, methodNames)](https://lodash.com/docs#bindAll) in lodash. ```` +## Partial functions + +Until now we have only been talking about binding `this`. Let's take it a step further. + +We can bind not only `this`, but also arguments. That's rarely done, but sometimes can be handy. + +The full syntax of `bind`: + +```js +let bound = func.bind(context, [arg1], [arg2], ...); +``` + +It allows to bind context as `this` and starting arguments of the function. + +For instance, we have a multiplication function `mul(a, b)`: + +```js +function mul(a, b) { + return a * b; +} +``` + +Let's use `bind` to create a function `double` on its base: + +```js run +function mul(a, b) { + return a * b; +} + +*!* +let double = mul.bind(null, 2); +*/!* + +alert( double(3) ); // = mul(2, 3) = 6 +alert( double(4) ); // = mul(2, 4) = 8 +alert( double(5) ); // = mul(2, 5) = 10 +``` + +The call to `mul.bind(null, 2)` creates a new function `double` that passes calls to `mul`, fixing `null` as the context and `2` as the first argument. Further arguments are passed "as is". + +That's called [partial function application](https://en.wikipedia.org/wiki/Partial_application) -- we create a new function by fixing some parameters of the existing one. + +Please note that we actually don't use `this` here. But `bind` requires it, so we must put in something like `null`. + +The function `triple` in the code below triples the value: + +```js run +function mul(a, b) { + return a * b; +} + +*!* +let triple = mul.bind(null, 3); +*/!* + +alert( triple(3) ); // = mul(3, 3) = 9 +alert( triple(4) ); // = mul(3, 4) = 12 +alert( triple(5) ); // = mul(3, 5) = 15 +``` + +Why do we usually make a partial function? + +The benefit is that we can create an independent function with a readable name (`double`, `triple`). We can use it and not provide the first argument every time as it's fixed with `bind`. + +In other cases, partial application is useful when we have a very generic function and want a less universal variant of it for convenience. + +For instance, we have a function `send(from, to, text)`. Then, inside a `user` object we may want to use a partial variant of it: `sendTo(to, text)` that sends from the current user. + +## Going partial without context + +What if we'd like to fix some arguments, but not the context `this`? For example, for an object method. + +The native `bind` does not allow that. We can't just omit the context and jump to arguments. + +Fortunately, a function `partial` for binding only arguments can be easily implemented. + +Like this: + +```js run +*!* +function partial(func, ...argsBound) { + return function(...args) { // (*) + return func.call(this, ...argsBound, ...args); + } +} +*/!* + +// Usage: +let user = { + firstName: "John", + say(time, phrase) { + alert(`[${time}] ${this.firstName}: ${phrase}!`); + } +}; + +// add a partial method with fixed time +user.sayNow = partial(user.say, new Date().getHours() + ':' + new Date().getMinutes()); + +user.sayNow("Hello"); +// Something like: +// [10:00] John: Hello! +``` + +The result of `partial(func[, arg1, arg2...])` call is a wrapper `(*)` that calls `func` with: +- Same `this` as it gets (for `user.sayNow` call it's `user`) +- Then gives it `...argsBound` -- arguments from the `partial` call (`"10:00"`) +- Then gives it `...args` -- arguments given to the wrapper (`"Hello"`) + +So easy to do it with the spread syntax, right? + +Also there's a ready [_.partial](https://lodash.com/docs#partial) implementation from lodash library. + ## Summary Method `func.bind(context, ...args)` returns a "bound variant" of function `func` that fixes the context `this` and first arguments if given. -Usually we apply `bind` to fix `this` in an object method, so that we can pass it somewhere. For example, to `setTimeout`. There are more reasons to `bind` in the modern development, we'll meet them later. +Usually we apply `bind` to fix `this` for an object method, so that we can pass it somewhere. For example, to `setTimeout`. + +When we fix some arguments of an existing function, the resulting (less universal) function is called *partially applied* or *partial*. + +Partials are convenient when we don't want to repeat the same argument over and over again. Like if we have a `send(from, to)` function, and `from` should always be the same for our task, we can get a partial and go on with it. diff --git a/1-js/06-advanced-functions/11-currying-partials/article.md b/1-js/06-advanced-functions/11-currying-partials/article.md deleted file mode 100644 index b97e6a5fc..000000000 --- a/1-js/06-advanced-functions/11-currying-partials/article.md +++ /dev/null @@ -1,309 +0,0 @@ -libs: - - lodash - ---- - -# Currying and partials - -Until now we have only been talking about binding `this`. Let's take it a step further. - -We can bind not only `this`, but also arguments. That's rarely done, but sometimes can be handy. - -The full syntax of `bind`: - -```js -let bound = func.bind(context, arg1, arg2, ...); -``` - -It allows to bind context as `this` and starting arguments of the function. - -For instance, we have a multiplication function `mul(a, b)`: - -```js -function mul(a, b) { - return a * b; -} -``` - -Let's use `bind` to create a function `double` on its base: - -```js run -function mul(a, b) { - return a * b; -} - -*!* -let double = mul.bind(null, 2); -*/!* - -alert( double(3) ); // = mul(2, 3) = 6 -alert( double(4) ); // = mul(2, 4) = 8 -alert( double(5) ); // = mul(2, 5) = 10 -``` - -The call to `mul.bind(null, 2)` creates a new function `double` that passes calls to `mul`, fixing `null` as the context and `2` as the first argument. Further arguments are passed "as is". - -That's called [partial function application](https://en.wikipedia.org/wiki/Partial_application) -- we create a new function by fixing some parameters of the existing one. - -Please note that here we actually don't use `this` here. But `bind` requires it, so we must put in something like `null`. - -The function `triple` in the code below triples the value: - -```js run -function mul(a, b) { - return a * b; -} - -*!* -let triple = mul.bind(null, 3); -*/!* - -alert( triple(3) ); // = mul(3, 3) = 9 -alert( triple(4) ); // = mul(3, 4) = 12 -alert( triple(5) ); // = mul(3, 5) = 15 -``` - -Why do we usually make a partial function? - -The benefit is that we can create an independent function with a readable name (`double`, `triple`). We can use it and not provide first argument of every time as it's fixed with `bind`. - -In other cases, partial application is useful when we have a very generic function and want a less universal variant of it for convenience. - -For instance, we have a function `send(from, to, text)`. Then, inside a `user` object we may want to use a partial variant of it: `sendTo(to, text)` that sends from the current user. - -## Going partial without context - -What if we'd like to fix some arguments, but not bind `this`? - -The native `bind` does not allow that. We can't just omit the context and jump to arguments. - -Fortunately, a `partial` function for binding only arguments can be easily implemented. - -Like this: - -```js run -*!* -function partial(func, ...argsBound) { - return function(...args) { // (*) - return func.call(this, ...argsBound, ...args); - } -} -*/!* - -// Usage: -let user = { - firstName: "John", - say(time, phrase) { - alert(`[${time}] ${this.firstName}: ${phrase}!`); - } -}; - -// add a partial method that says something now by fixing the first argument -user.sayNow = partial(user.say, new Date().getHours() + ':' + new Date().getMinutes()); - -user.sayNow("Hello"); -// Something like: -// [10:00] John: Hello! -``` - -The result of `partial(func[, arg1, arg2...])` call is a wrapper `(*)` that calls `func` with: -- Same `this` as it gets (for `user.sayNow` call it's `user`) -- Then gives it `...argsBound` -- arguments from the `partial` call (`"10:00"`) -- Then gives it `...args` -- arguments given to the wrapper (`"Hello"`) - -So easy to do it with the spread operator, right? - -Also there's a ready [_.partial](https://lodash.com/docs#partial) implementation from lodash library. - -## Currying - -Sometimes people mix up partial function application mentioned above with another thing named "currying". That's another interesting technique of working with functions that we just have to mention here. - -[Currying](https://en.wikipedia.org/wiki/Currying) is a transformation of functions that translates a function from callable as `f(a, b, c)` into callable as `f(a)(b)(c)`. In JavaScript, we usually make a wrapper to keep the original function. - -Currying doesn't call a function. It just transforms it. - -Let's create a helper `curry(f)` function that performs currying for a two-argument `f`. In other words, `curry(f)` for two-argument `f(a, b)` translates it into `f(a)(b)` - -```js run -*!* -function curry(f) { // curry(f) does the currying transform - return function(a) { - return function(b) { - return f(a, b); - }; - }; -} -*/!* - -// usage -function sum(a, b) { - return a + b; -} - -let carriedSum = curry(sum); - -alert( carriedSum(1)(2) ); // 3 -``` - -As you can see, the implementation is a series of wrappers. - -- The result of `curry(func)` is a wrapper `function(a)`. -- When it is called like `sum(1)`, the argument is saved in the Lexical Environment, and a new wrapper is returned `function(b)`. -- Then `sum(1)(2)` finally calls `function(b)` providing `2`, and it passes the call to the original multi-argument `sum`. - -More advanced implementations of currying like [_.curry](https://lodash.com/docs#curry) from lodash library do something more sophisticated. They return a wrapper that allows a function to be called normally when all arguments are supplied *or* returns a partial otherwise. - -```js -function curry(f) { - return function(...args) { - // if args.length == f.length (as many arguments as f has), - // then pass the call to f - // otherwise return a partial function that fixes args as first arguments - }; -} -``` - -## Currying? What for? - -To understand the benefits we definitely need a worthy real-life example. - -Advanced currying allows the function to be both callable normally and partially. - -For instance, we have the logging function `log(date, importance, message)` that formats and outputs the information. In real projects such functions also have many other useful features like sending logs over the network: - -```js -function log(date, importance, message) { - alert(`[${date.getHours()}:${date.getMinutes()}] [${importance}] ${message}`); -} -``` - -Let's curry it! - -```js -log = _.curry(log); -``` - -After that `log` still works the normal way: - -```js -log(new Date(), "DEBUG", "some debug"); -``` - -...But also can be called in the curried form: - -```js -log(new Date())("DEBUG")("some debug"); // log(a)(b)(c) -``` - -Let's get a convenience function for today's logs: - -```js -// todayLog will be the partial of log with fixed first argument -let todayLog = log(new Date()); - -// use it -todayLog("INFO", "message"); // [HH:mm] INFO message -``` - -And now a convenience function for today's debug messages: - -```js -let todayDebug = todayLog("DEBUG"); - -todayDebug("message"); // [HH:mm] DEBUG message -``` - -So: -1. We didn't lose anything after currying: `log` is still callable normally. -2. We were able to generate partial functions such as for today's logs. - -## Advanced curry implementation - -In case you'd like to get in details (not obligatory!), here's the "advanced" curry implementation that we could use above. - -It's pretty short: - -```js -function curry(func) { - - return function curried(...args) { - if (args.length >= func.length) { - return func.apply(this, args); - } else { - return function(...args2) { - return curried.apply(this, args.concat(args2)); - } - } - }; - -} -``` - -Usage examples: - -```js -function sum(a, b, c) { - return a + b + c; -} - -let curriedSum = curry(sum); - -alert( curriedSum(1, 2, 3) ); // 6, still callable normally -alert( curriedSum(1)(2,3) ); // 6, currying of 1st arg -alert( curriedSum(1)(2)(3) ); // 6, full currying -``` - -The new `curry` may look complicated, but it's actually easy to understand. - -The result of `curry(func)` is the wrapper `curried` that looks like this: - -```js -// func is the function to transform -function curried(...args) { - if (args.length >= func.length) { // (1) - return func.apply(this, args); - } else { - return function pass(...args2) { // (2) - return curried.apply(this, args.concat(args2)); - } - } -}; -``` - -When we run it, there are two branches: - -1. Call now: if passed `args` count is the same as the original function has in its definition (`func.length`) or longer, then just pass the call to it. -2. Get a partial: otherwise, `func` is not called yet. Instead, another wrapper `pass` is returned, that will re-apply `curried` providing previous arguments together with the new ones. Then on a new call, again, we'll get either a new partial (if not enough arguments) or, finally, the result. - -For instance, let's see what happens in the case of `sum(a, b, c)`. Three arguments, so `sum.length = 3`. - -For the call `curried(1)(2)(3)`: - -1. The first call `curried(1)` remembers `1` in its Lexical Environment, and returns a wrapper `pass`. -2. The wrapper `pass` is called with `(2)`: it takes previous args (`1`), concatenates them with what it got `(2)` and calls `curried(1, 2)` with them together. - - As the argument count is still less than 3, `curry` returns `pass`. -3. The wrapper `pass` is called again with `(3)`, for the next call `pass(3)` takes previous args (`1`, `2`) and adds `3` to them, making the call `curried(1, 2, 3)` -- there are `3` arguments at last, they are given to the original function. - -If that's still not obvious, just trace the calls sequence in your mind or on the paper. - -```smart header="Fixed-length functions only" -The currying requires the function to have a known fixed number of arguments. -``` - -```smart header="A little more than currying" -By definition, currying should convert `sum(a, b, c)` into `sum(a)(b)(c)`. - -But most implementations of currying in JavaScript are advanced, as described: they also keep the function callable in the multi-argument variant. -``` - -## Summary - -- When we fix some arguments of an existing function, the resulting (less universal) function is called *a partial*. We can use `bind` to get a partial, but there are other ways also. - - Partials are convenient when we don't want to repeat the same argument over and over again. Like if we have a `send(from, to)` function, and `from` should always be the same for our task, we can get a partial and go on with it. - -- *Currying* is a transform that makes `f(a,b,c)` callable as `f(a)(b)(c)`. JavaScript implementations usually both keep the function callable normally and return the partial if arguments count is not enough. - - Currying is great when we want easy partials. As we've seen in the logging example: the universal function `log(date, importance, message)` after currying gives us partials when called with one argument like `log(date)` or two arguments `log(date, importance)`. diff --git a/1-js/06-advanced-functions/12-arrow-functions/article.md b/1-js/06-advanced-functions/12-arrow-functions/article.md index 1ade1a419..8730277ad 100644 --- a/1-js/06-advanced-functions/12-arrow-functions/article.md +++ b/1-js/06-advanced-functions/12-arrow-functions/article.md @@ -2,9 +2,9 @@ Let's revisit arrow functions. -Arrow functions are not just a "shorthand" for writing small stuff. +Arrow functions are not just a "shorthand" for writing small stuff. They have some very specific and useful features. -JavaScript is full of situations where we need to write a small function, that's executed somewhere else. +JavaScript is full of situations where we need to write a small function that's executed somewhere else. For instance: @@ -14,7 +14,7 @@ For instance: It's in the very spirit of JavaScript to create a function and pass it somewhere. -And in such functions we usually don't want to leave the current context. +And in such functions we usually don't want to leave the current context. That's where arrow functions come in handy. ## Arrow functions have no "this" @@ -52,7 +52,7 @@ let group = { *!* this.students.forEach(function(student) { // Error: Cannot read property 'title' of undefined - alert(this.title + ': ' + student) + alert(this.title + ': ' + student); }); */!* } @@ -87,7 +87,7 @@ For instance, `defer(f, ms)` gets a function and returns a wrapper around it tha ```js run function defer(f, ms) { return function() { - setTimeout(() => f.apply(this, arguments), ms) + setTimeout(() => f.apply(this, arguments), ms); }; } @@ -118,9 +118,9 @@ Here we had to create additional variables `args` and `ctx` so that the function Arrow functions: -- Do not have `this`. -- Do not have `arguments`. -- Can't be called with `new`. -- (They also don't have `super`, but we didn't study it. Will be in the chapter <info:class-inheritance>). +- Do not have `this` +- Do not have `arguments` +- Can't be called with `new` +- They also don't have `super`, but we didn't study it yet. We will on the chapter <info:class-inheritance> -That's because they are meant for short pieces of code that do not have their own "context", but rather works in the current one. And they really shine in that use case. +That's because they are meant for short pieces of code that do not have their own "context", but rather work in the current one. And they really shine in that use case. diff --git a/1-js/07-object-properties/01-property-descriptors/article.md b/1-js/07-object-properties/01-property-descriptors/article.md index 7768b3557..0a945b377 100644 --- a/1-js/07-object-properties/01-property-descriptors/article.md +++ b/1-js/07-object-properties/01-property-descriptors/article.md @@ -3,7 +3,7 @@ As we know, objects can store properties. -Till now, a property was a simple "key-value" pair to us. But an object property is actually a more flexible and powerful thing. +Until now, a property was a simple "key-value" pair to us. But an object property is actually a more flexible and powerful thing. In this chapter we'll study additional configuration options, and in the next we'll see how to invisibly turn them into getter/setter functions. @@ -11,7 +11,7 @@ In this chapter we'll study additional configuration options, and in the next we Object properties, besides a **`value`**, have three special attributes (so-called "flags"): -- **`writable`** -- if `true`, can be changed, otherwise it's read-only. +- **`writable`** -- if `true`, the value can be changed, otherwise it's read-only. - **`enumerable`** -- if `true`, then listed in loops, otherwise not listed. - **`configurable`** -- if `true`, the property can be deleted and these attributes can be modified, otherwise not. @@ -19,7 +19,7 @@ We didn't see them yet, because generally they do not show up. When we create a First, let's see how to get those flags. -The method [Object.getOwnPropertyDescriptor](mdn:js/Object/getOwnPropertyDescriptor) allows to query the *full* information about a property. +The method [Object.getOwnPropertyDescriptor](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/getOwnPropertyDescriptor) allows to query the *full* information about a property. The syntax is: ```js @@ -54,7 +54,7 @@ alert( JSON.stringify(descriptor, null, 2 ) ); */ ``` -To change the flags, we can use [Object.defineProperty](mdn:js/Object/defineProperty). +To change the flags, we can use [Object.defineProperty](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty). The syntax is: @@ -63,10 +63,10 @@ Object.defineProperty(obj, propertyName, descriptor) ``` `obj`, `propertyName` -: The object and property to work on. +: The object and its property to apply the descriptor. `descriptor` -: Property descriptor to apply. +: Property descriptor object to apply. If the property exists, `defineProperty` updates its flags. Otherwise, it creates the property with the given value and flags; in that case, if a flag is not supplied, it is assumed `false`. @@ -100,9 +100,9 @@ Compare it with "normally created" `user.name` above: now all flags are falsy. I Now let's see effects of the flags by example. -## Read-only +## Non-writable -Let's make `user.name` read-only by changing `writable` flag: +Let's make `user.name` non-writable (can't be reassigned) by changing `writable` flag: ```js run let user = { @@ -116,36 +116,39 @@ Object.defineProperty(user, "name", { }); *!* -user.name = "Pete"; // Error: Cannot assign to read only property 'name'... +user.name = "Pete"; // Error: Cannot assign to read only property 'name' */!* ``` Now no one can change the name of our user, unless they apply their own `defineProperty` to override ours. -Here's the same operation, but for the case when a property doesn't exist: +```smart header="Errors appear only in strict mode" +In non-strict mode, no errors occur when writing to non-writable properties and such. But the operation still won't succeed. Flag-violating actions are just silently ignored in non-strict. +``` + +Here's the same example, but the property is created from scratch: ```js run let user = { }; Object.defineProperty(user, "name", { *!* - value: "Pete", - // for new properties need to explicitly list what's true + value: "John", + // for new properties we need to explicitly list what's true enumerable: true, configurable: true */!* }); -alert(user.name); // Pete -user.name = "Alice"; // Error +alert(user.name); // John +user.name = "Pete"; // Error ``` - ## Non-enumerable Now let's add a custom `toString` to `user`. -Normally, a built-in `toString` for objects is non-enumerable, it does not show up in `for..in`. But if we add `toString` of our own, then by default it shows up in `for..in`, like this: +Normally, a built-in `toString` for objects is non-enumerable, it does not show up in `for..in`. But if we add a `toString` of our own, then by default it shows up in `for..in`, like this: ```js run let user = { @@ -159,7 +162,7 @@ let user = { for (let key in user) alert(key); // name, toString ``` -If we don't like it, then we can set `enumerable:false`. Then it won't appear in `for..in` loop, just like the built-in one: +If we don't like it, then we can set `enumerable:false`. Then it won't appear in a `for..in` loop, just like the built-in one: ```js run let user = { @@ -191,9 +194,9 @@ alert(Object.keys(user)); // name The non-configurable flag (`configurable:false`) is sometimes preset for built-in objects and properties. -A non-configurable property can not be deleted or altered with `defineProperty`. +A non-configurable property can't be deleted, its attributes can't be modified. -For instance, `Math.PI` is read-only, non-enumerable and non-configurable: +For instance, `Math.PI` is non-writable, non-enumerable and non-configurable: ```js run let descriptor = Object.getOwnPropertyDescriptor(Math, 'PI'); @@ -211,41 +214,67 @@ alert( JSON.stringify(descriptor, null, 2 ) ); So, a programmer is unable to change the value of `Math.PI` or overwrite it. ```js run -Math.PI = 3; // Error +Math.PI = 3; // Error, because it has writable: false // delete Math.PI won't work either ``` -Making a property non-configurable is a one-way road. We cannot change it back, because `defineProperty` doesn't work on non-configurable properties. +We also can't change `Math.PI` to be `writable` again: + +```js run +// Error, because of configurable: false +Object.defineProperty(Math, "PI", { writable: true }); +``` + +There's absolutely nothing we can do with `Math.PI`. -Here we are making `user.name` a "forever sealed" constant: +Making a property non-configurable is a one-way road. We cannot change it back with `defineProperty`. + +**Please note: `configurable: false` prevents changes of property flags and its deletion, while allowing to change its value.** + +Here `user.name` is non-configurable, but we can still change it (as it's writable): ```js run -let user = { }; +let user = { + name: "John" +}; + +Object.defineProperty(user, "name", { + configurable: false +}); + +user.name = "Pete"; // works fine +delete user.name; // Error +``` + +And here we make `user.name` a "forever sealed" constant, just like the built-in `Math.PI`: + +```js run +let user = { + name: "John" +}; Object.defineProperty(user, "name", { - value: "John", writable: false, configurable: false }); -*!* // won't be able to change user.name or its flags // all this won't work: -// user.name = "Pete" -// delete user.name -// defineProperty(user, "name", ...) -Object.defineProperty(user, "name", {writable: true}); // Error -*/!* +user.name = "Pete"; +delete user.name; +Object.defineProperty(user, "name", { value: "Pete" }); ``` -```smart header="Errors appear only in use strict" -In the non-strict mode, no errors occur when writing to read-only properties and such. But the operation still won't succeed. Flag-violating actions are just silently ignored in non-strict. +```smart header="The only attribute change possible: writable true -> false" +There's a minor exception about changing flags. + +We can change `writable: true` to `false` for a non-configurable property, thus preventing its value modification (to add another layer of protection). Not the other way around though. ``` ## Object.defineProperties -There's a method [Object.defineProperties(obj, descriptors)](mdn:js/Object/defineProperties) that allows to define many properties at once. +There's a method [Object.defineProperties(obj, descriptors)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperties) that allows to define many properties at once. The syntax is: @@ -271,7 +300,7 @@ So, we can set many properties at once. ## Object.getOwnPropertyDescriptors -To get all property descriptors at once, we can use the method [Object.getOwnPropertyDescriptors(obj)](mdn:js/Object/getOwnPropertyDescriptors). +To get all property descriptors at once, we can use the method [Object.getOwnPropertyDescriptors(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/getOwnPropertyDescriptors). Together with `Object.defineProperties` it can be used as a "flags-aware" way of cloning an object: @@ -289,7 +318,7 @@ for (let key in user) { ...But that does not copy flags. So if we want a "better" clone then `Object.defineProperties` is preferred. -Another difference is that `for..in` ignores symbolic properties, but `Object.getOwnPropertyDescriptors` returns *all* property descriptors including symbolic ones. +Another difference is that `for..in` ignores symbolic and non-enumerable properties, but `Object.getOwnPropertyDescriptors` returns *all* property descriptors including symbolic and non-enumerable ones. ## Sealing an object globally @@ -297,23 +326,24 @@ Property descriptors work at the level of individual properties. There are also methods that limit access to the *whole* object: -[Object.preventExtensions(obj)](mdn:js/Object/preventExtensions) +[Object.preventExtensions(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/preventExtensions) : Forbids the addition of new properties to the object. -[Object.seal(obj)](mdn:js/Object/seal) +[Object.seal(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/seal) : Forbids adding/removing of properties. Sets `configurable: false` for all existing properties. -[Object.freeze(obj)](mdn:js/Object/freeze) +[Object.freeze(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/freeze) : Forbids adding/removing/changing of properties. Sets `configurable: false, writable: false` for all existing properties. + And also there are tests for them: -[Object.isExtensible(obj)](mdn:js/Object/isExtensible) +[Object.isExtensible(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/isExtensible) : Returns `false` if adding properties is forbidden, otherwise `true`. -[Object.isSealed(obj)](mdn:js/Object/isSealed) +[Object.isSealed(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/isSealed) : Returns `true` if adding/removing properties is forbidden, and all existing properties have `configurable: false`. -[Object.isFrozen(obj)](mdn:js/Object/isFrozen) +[Object.isFrozen(obj)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/isFrozen) : Returns `true` if adding/removing/changing properties is forbidden, and all current properties are `configurable: false, writable: false`. These methods are rarely used in practice. diff --git a/1-js/07-object-properties/02-property-accessors/article.md b/1-js/07-object-properties/02-property-accessors/article.md index 43cd5ae6d..c2aa35d53 100644 --- a/1-js/07-object-properties/02-property-accessors/article.md +++ b/1-js/07-object-properties/02-property-accessors/article.md @@ -1,11 +1,11 @@ # Property getters and setters -There are two kinds of properties. +There are two kinds of object properties. -The first kind is *data properties*. We already know how to work with them. Actually, all properties that we've been using till now were data properties. +The first kind is *data properties*. We already know how to work with them. All properties that we've been using until now were data properties. -The second type of properties is something new. It's *accessor properties*. They are essentially functions that work on getting and setting a value, but look like regular properties to an external code. +The second type of property is something new. It's an *accessor property*. They are essentially functions that execute on getting and setting a value, but look like regular properties to an external code. ## Getters and setters @@ -27,14 +27,14 @@ The getter works when `obj.propName` is read, the setter -- when it is assigned. For instance, we have a `user` object with `name` and `surname`: -```js run +```js let user = { name: "John", surname: "Smith" }; ``` -Now we want to add a "fullName" property, that should be "John Smith". Of course, we don't want to copy-paste existing information, so we can implement it as an accessor: +Now we want to add a `fullName` property, that should be `"John Smith"`. Of course, we don't want to copy-paste existing information, so we can implement it as an accessor: ```js run let user = { @@ -53,9 +53,21 @@ alert(user.fullName); // John Smith */!* ``` -From outside, an accessor property looks like a regular one. That's the idea of accessor properties. We don't *call* `user.fullName` as a function, we *read* it normally: the getter runs behind the scenes. +From the outside, an accessor property looks like a regular one. That's the idea of accessor properties. We don't *call* `user.fullName` as a function, we *read* it normally: the getter runs behind the scenes. + +As of now, `fullName` has only a getter. If we attempt to assign `user.fullName=`, there will be an error: -As of now, `fullName` has only a getter. If we attempt to assign `user.fullName=`, there will be an error. +```js run +let user = { + get fullName() { + return `...`; + } +}; + +*!* +user.fullName = "Test"; // Error (property has only a getter) +*/!* +``` Let's fix it by adding a setter for `user.fullName`: @@ -82,25 +94,15 @@ alert(user.name); // Alice alert(user.surname); // Cooper ``` -Now we have a "virtual" property. It is readable and writable, but in fact does not exist. - -```smart header="Accessor properties are only accessible with get/set" -Once a property is defined with `get prop()` or `set prop()`, it's an accessor property, not a data properety any more. - -- If there's a getter -- we can read `object.prop`, othrewise we can't. -- If there's a setter -- we can set `object.prop=...`, othrewise we can't. - -And in either case we can't `delete` an accessor property. -``` - +As the result, we have a "virtual" property `fullName`. It is readable and writable. ## Accessor descriptors -Descriptors for accessor properties are different -- as compared with data properties. +Descriptors for accessor properties are different from those for data properties. -For accessor properties, there is no `value` and `writable`, but instead there are `get` and `set` functions. +For accessor properties, there is no `value` or `writable`, but instead there are `get` and `set` functions. -So an accessor descriptor may have: +That is, an accessor descriptor may have: - **`get`** -- a function without arguments, that works when a property is read, - **`set`** -- a function with one argument, that is called when the property is set, @@ -132,7 +134,7 @@ alert(user.fullName); // John Smith for(let key in user) alert(key); // name, surname ``` -Please note once again that a property can be either an accessor or a data property, not both. +Please note that a property can be either an accessor (has `get/set` methods) or a data property (has a `value`), not both. If we try to supply both `get` and `value` in the same descriptor, there will be an error: @@ -151,9 +153,9 @@ Object.defineProperty({}, 'prop', { ## Smarter getters/setters -Getters/setters can be used as wrappers over "real" property values to gain more control over them. +Getters/setters can be used as wrappers over "real" property values to gain more control over operations with them. -For instance, if we want to forbid too short names for `user`, we can store `name` in a special property `_name`. And filter assignments in the setter: +For instance, if we want to forbid too short names for `user`, we can have a setter `name` and keep the value in a separate property `_name`: ```js run let user = { @@ -176,14 +178,16 @@ alert(user.name); // Pete user.name = ""; // Name is too short... ``` -Technically, the external code may still access the name directly by using `user._name`. But there is a widely known agreement that properties starting with an underscore `"_"` are internal and should not be touched from outside the object. +So, the name is stored in `_name` property, and the access is done via getter and setter. + +Technically, external code is able to access the name directly by using `user._name`. But there is a widely known convention that properties starting with an underscore `"_"` are internal and should not be touched from outside the object. ## Using for compatibility -One of the great ideas behind getters and setters -- they allow to take control over a "normal" data property and tweak it at any moment. +One of the great uses of accessors is that they allow to take control over a "regular" data property at any moment by replacing it with a getter and a setter and tweak its behavior. -For instance, we started implementing user objects using data properties `name` and `age`: +Imagine we started implementing user objects using data properties `name` and `age`: ```js function User(name, age) { @@ -209,9 +213,11 @@ let john = new User("John", new Date(1992, 6, 1)); Now what to do with the old code that still uses `age` property? -We can try to find all such places and fix them, but that takes time and can be hard to do if that code is written by other people. And besides, `age` is a nice thing to have in `user`, right? In some places it's just what we want. +We can try to find all such places and fix them, but that takes time and can be hard to do if that code is used by many other people. And besides, `age` is a nice thing to have in `user`, right? + +Let's keep it. -Adding a getter for `age` mitigates the problem: +Adding a getter for `age` solves the problem: ```js run no-beautify function User(name, birthday) { diff --git a/1-js/08-prototypes/01-prototype-inheritance/2-search-algorithm/task.md b/1-js/08-prototypes/01-prototype-inheritance/2-search-algorithm/task.md index 002b24b8a..bc2db47fe 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/2-search-algorithm/task.md +++ b/1-js/08-prototypes/01-prototype-inheritance/2-search-algorithm/task.md @@ -6,7 +6,7 @@ importance: 5 The task has two parts. -We have an object: +Given the following objects: ```js let head = { diff --git a/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/solution.md b/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/solution.md index c7d147b9c..4d6ea2653 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/solution.md +++ b/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/solution.md @@ -3,4 +3,5 @@ That's because `this` is an object before the dot, so `rabbit.eat()` modifies `rabbit`. Property lookup and execution are two different things. -The method `rabbit.eat` is first found in the prototype, then executed with `this=rabbit` + +The method `rabbit.eat` is first found in the prototype, then executed with `this=rabbit`. diff --git a/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/task.md b/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/task.md index b37499bad..ed8482c07 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/task.md +++ b/1-js/08-prototypes/01-prototype-inheritance/3-proto-and-this/task.md @@ -2,7 +2,7 @@ importance: 5 --- -# Where it writes? +# Where does it write? We have `rabbit` inheriting from `animal`. diff --git a/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/solution.md b/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/solution.md index fad4b8860..c141b2ecd 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/solution.md +++ b/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/solution.md @@ -10,7 +10,7 @@ Let's look carefully at what's going on in the call `speedy.eat("apple")`. So all hamsters share a single stomach! -Every time the `stomach` is taken from the prototype, then `stomach.push` modifies it "at place". +Both for `lazy.stomach.push(...)` and `speedy.stomach.push()`, the property `stomach` is found in the prototype (as it's not in the object itself), then the new data is pushed into it. Please note that such thing doesn't happen in case of a simple assignment `this.stomach=`: @@ -44,7 +44,7 @@ alert( lazy.stomach ); // <nothing> Now all works fine, because `this.stomach=` does not perform a lookup of `stomach`. The value is written directly into `this` object. -Also we can totally evade the problem by making sure that each hamster has their own stomach: +Also we can totally avoid the problem by making sure that each hamster has their own stomach: ```js run let hamster = { @@ -77,4 +77,4 @@ alert( speedy.stomach ); // apple alert( lazy.stomach ); // <nothing> ``` -As a common solution, all properties that describe the state of a particular object, like `stomach` above, are usually written into that object. That prevents such problems. +As a common solution, all properties that describe the state of a particular object, like `stomach` above, should be written into that object. That prevents such problems. diff --git a/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/task.md b/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/task.md index 6f9fb279e..50171123d 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/task.md +++ b/1-js/08-prototypes/01-prototype-inheritance/4-hamster-proto/task.md @@ -2,11 +2,11 @@ importance: 5 --- -# Why two hamsters are full? +# Why are both hamsters full? We have two hamsters: `speedy` and `lazy` inheriting from the general `hamster` object. -When we feed one of them, the other one is also full. Why? How to fix it? +When we feed one of them, the other one is also full. Why? How can we fix it? ```js run let hamster = { diff --git a/1-js/08-prototypes/01-prototype-inheritance/article.md b/1-js/08-prototypes/01-prototype-inheritance/article.md index 7c106b1f7..ef6c7ffeb 100644 --- a/1-js/08-prototypes/01-prototype-inheritance/article.md +++ b/1-js/08-prototypes/01-prototype-inheritance/article.md @@ -12,11 +12,11 @@ In JavaScript, objects have a special hidden property `[[Prototype]]` (as named  -That `[[Prototype]]` has a "magical" meaning. When we want to read a property from `object`, and it's missing, JavaScript automatically takes it from the prototype. In programming, such thing is called "prototypal inheritance". Many cool language features and programming techniques are based on it. +When we read a property from `object`, and it's missing, JavaScript automatically takes it from the prototype. In programming, this is called "prototypal inheritance". And soon we'll study many examples of such inheritance, as well as cooler language features built upon it. The property `[[Prototype]]` is internal and hidden, but there are many ways to set it. -One of them is to use `__proto__`, like this: +One of them is to use the special name `__proto__`, like this: ```js run let animal = { @@ -27,23 +27,15 @@ let rabbit = { }; *!* -rabbit.__proto__ = animal; +rabbit.__proto__ = animal; // sets rabbit.[[Prototype]] = animal */!* ``` -```smart header="`__proto__` is a historical getter/setter for `[[Prototype]]`" -Please note that `__proto__` is *not the same* as `[[Prototype]]`. That's a getter/setter for it. - -It exists for historical reasons, in modern language it is replaced with functions `Object.getPrototypeOf/Object.setPrototypeOf` that also get/set the prototype. We'll study the reasons for that and these functions later. - -By the specification, `__proto__` must only be supported by browsers, but in fact all environments including server-side support it. For now, as `__proto__` notation is a little bit more intuitively obvious, we'll use it in the examples. -``` - -If we look for a property in `rabbit`, and it's missing, JavaScript automatically takes it from `animal`. +Now if we read a property from `rabbit`, and it's missing, JavaScript will automatically take it from `animal`. For instance: -```js run +```js let animal = { eats: true }; @@ -62,7 +54,7 @@ alert( rabbit.eats ); // true (**) alert( rabbit.jumps ); // true ``` -Here the line `(*)` sets `animal` to be a prototype of `rabbit`. +Here the line `(*)` sets `animal` to be the prototype of `rabbit`. Then, when `alert` tries to read property `rabbit.eats` `(**)`, it's not in `rabbit`, so JavaScript follows the `[[Prototype]]` reference and finds it in `animal` (look from the bottom up): @@ -101,7 +93,6 @@ The method is automatically taken from the prototype, like this: The prototype chain can be longer: - ```js run let animal = { eats: true, @@ -131,13 +122,27 @@ alert(longEar.jumps); // true (from rabbit)  -There are actually only two limitations: +Now if we read something from `longEar`, and it's missing, JavaScript will look for it in `rabbit`, and then in `animal`. + +There are only two limitations: 1. The references can't go in circles. JavaScript will throw an error if we try to assign `__proto__` in a circle. -2. The value of `__proto__` can be either an object or `null`, other types (like primitives) are ignored. +2. The value of `__proto__` can be either an object or `null`. Other types are ignored. Also it may be obvious, but still: there can be only one `[[Prototype]]`. An object may not inherit from two others. +```smart header="`__proto__` is a historical getter/setter for `[[Prototype]]`" +It's a common mistake of novice developers not to know the difference between these two. + +Please note that `__proto__` is *not the same* as the internal `[[Prototype]]` property. It's a getter/setter for `[[Prototype]]`. Later we'll see situations where it matters, for now let's just keep it in mind, as we build our understanding of JavaScript language. + +The `__proto__` property is a bit outdated. It exists for historical reasons, modern JavaScript suggests that we should use `Object.getPrototypeOf/Object.setPrototypeOf` functions instead that get/set the prototype. We'll also cover these functions later. + +By the specification, `__proto__` must only be supported by browsers. In fact though, all environments including server-side support `__proto__`, so we're quite safe using it. + +As the `__proto__` notation is a bit more intuitively obvious, we use it in the examples. +``` + ## Writing doesn't use prototype The prototype is only used for reading properties. @@ -171,7 +176,7 @@ From now on, `rabbit.walk()` call finds the method immediately in the object and  -That's for data properties only, not for accessors. If a property is a getter/setter, then it behaves like a function: getters/setters are looked up in the prototype. +Accessor properties are an exception, as assignment is handled by a setter function. So writing to such a property is actually the same as calling a function. For that reason `admin.fullName` works correctly in the code below: @@ -198,13 +203,16 @@ alert(admin.fullName); // John Smith (*) // setter triggers! admin.fullName = "Alice Cooper"; // (**) + +alert(admin.fullName); // Alice Cooper, state of admin modified +alert(user.fullName); // John Smith, state of user protected ``` Here in the line `(*)` the property `admin.fullName` has a getter in the prototype `user`, so it is called. And in the line `(**)` the property has a setter in the prototype, so it is called. ## The value of "this" -An interesting question may arise in the example above: what's the value of `this` inside `set fullName(value)`? Where the properties `this.name` and `this.surname` are written: into `user` or `admin`? +An interesting question may arise in the example above: what's the value of `this` inside `set fullName(value)`? Where are the properties `this.name` and `this.surname` written: into `user` or `admin`? The answer is simple: `this` is not affected by prototypes at all. @@ -212,7 +220,7 @@ The answer is simple: `this` is not affected by prototypes at all. So, the setter call `admin.fullName=` uses `admin` as `this`, not `user`. -That is actually a super-important thing, because we may have a big object with many methods and inherit from it. Then inherited objects can run its methods, and they will modify the state of these objects, not the big one. +That is actually a super-important thing, because we may have a big object with many methods, and have objects that inherit from it. And when the inheriting objects run the inherited methods, they will modify only their own states, not the state of the big object. For instance, here `animal` represents a "method storage", and `rabbit` makes use of it. @@ -247,14 +255,84 @@ The resulting picture:  -If we had other objects like `bird`, `snake` etc inheriting from `animal`, they would also gain access to methods of `animal`. But `this` in each method would be the corresponding object, evaluated at the call-time (before dot), not `animal`. So when we write data into `this`, it is stored into these objects. +If we had other objects, like `bird`, `snake`, etc., inheriting from `animal`, they would also gain access to methods of `animal`. But `this` in each method call would be the corresponding object, evaluated at the call-time (before dot), not `animal`. So when we write data into `this`, it is stored into these objects. As a result, methods are shared, but the object state is not. +## for..in loop + +The `for..in` loop iterates over inherited properties too. + +For instance: + +```js run +let animal = { + eats: true +}; + +let rabbit = { + jumps: true, + __proto__: animal +}; + +*!* +// Object.keys only returns own keys +alert(Object.keys(rabbit)); // jumps +*/!* + +*!* +// for..in loops over both own and inherited keys +for(let prop in rabbit) alert(prop); // jumps, then eats +*/!* +``` + +If that's not what we want, and we'd like to exclude inherited properties, there's a built-in method [obj.hasOwnProperty(key)](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/hasOwnProperty): it returns `true` if `obj` has its own (not inherited) property named `key`. + +So we can filter out inherited properties (or do something else with them): + +```js run +let animal = { + eats: true +}; + +let rabbit = { + jumps: true, + __proto__: animal +}; + +for(let prop in rabbit) { + let isOwn = rabbit.hasOwnProperty(prop); + + if (isOwn) { + alert(`Our: ${prop}`); // Our: jumps + } else { + alert(`Inherited: ${prop}`); // Inherited: eats + } +} +``` + +Here we have the following inheritance chain: `rabbit` inherits from `animal`, that inherits from `Object.prototype` (because `animal` is a literal object `{...}`, so it's by default), and then `null` above it: + + + +Note, there's one funny thing. Where is the method `rabbit.hasOwnProperty` coming from? We did not define it. Looking at the chain we can see that the method is provided by `Object.prototype.hasOwnProperty`. In other words, it's inherited. + +...But why does `hasOwnProperty` not appear in the `for..in` loop like `eats` and `jumps` do, if `for..in` lists inherited properties? + +The answer is simple: it's not enumerable. Just like all other properties of `Object.prototype`, it has `enumerable:false` flag. And `for..in` only lists enumerable properties. That's why it and the rest of the `Object.prototype` properties are not listed. + +```smart header="Almost all other key/value-getting methods ignore inherited properties" +Almost all other key/value-getting methods, such as `Object.keys`, `Object.values` and so on ignore inherited properties. + +They only operate on the object itself. Properties from the prototype are *not* taken into account. +``` + ## Summary - In JavaScript, all objects have a hidden `[[Prototype]]` property that's either another object or `null`. - We can use `obj.__proto__` to access it (a historical getter/setter, there are other ways, to be covered soon). - The object referenced by `[[Prototype]]` is called a "prototype". -- If we want to read a property of `obj` or call a method, and it doesn't exist, then JavaScript tries to find it in the prototype. Write/delete operations work directly on the object, they don't use the prototype (unless the property is actually a setter). +- If we want to read a property of `obj` or call a method, and it doesn't exist, then JavaScript tries to find it in the prototype. +- Write/delete operations act directly on the object, they don't use the prototype (assuming it's a data property, not a setter). - If we call `obj.method()`, and the `method` is taken from the prototype, `this` still references `obj`. So methods always work with the current object even if they are inherited. +- The `for..in` loop iterates over both its own and its inherited properties. All other key/value-getting methods only operate on the object itself. diff --git a/1-js/08-prototypes/01-prototype-inheritance/rabbit-animal-object.svg b/1-js/08-prototypes/01-prototype-inheritance/rabbit-animal-object.svg new file mode 100644 index 000000000..32a9858f8 --- /dev/null +++ b/1-js/08-prototypes/01-prototype-inheritance/rabbit-animal-object.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="233" height="344" viewBox="0 0 233 344"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="inheritance" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="rabbit-animal-object.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M14 96h218v58H14z"/><text id="toString:-function" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="24" y="113">toString: function</tspan> <tspan x="24" y="128">hasOwnProperty: function</tspan> <tspan x="24" y="143">...</tspan></text><text id="Object.prototype" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="88">Object.prototype</tspan></text><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M14 213h218v28H14z"/><text id="animal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="205">animal</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M73.5 162.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[Prototype]]" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="83" y="184">[[Prototype]]</tspan></text><text id="[[Prototype]]-Copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="83" y="274">[[Prototype]]</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M73.5 30.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[Prototype]]" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="83" y="52">[[Prototype]]</tspan></text><text id="null" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="59" y="19">null</tspan></text><text id="eats:-true" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="24" y="231">eats: true</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M14 302h218v28H14z"/><text id="rabbit-copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="294">rabbit</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M73.5 251.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="jumps:-true" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="24" y="320">jumps: true</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/08-prototypes/02-function-prototype/1-changing-prototype/solution.md b/1-js/08-prototypes/02-function-prototype/1-changing-prototype/solution.md index 771e3061c..ebbdf3a7c 100644 --- a/1-js/08-prototypes/02-function-prototype/1-changing-prototype/solution.md +++ b/1-js/08-prototypes/02-function-prototype/1-changing-prototype/solution.md @@ -7,7 +7,7 @@ Answers: 2. `false`. - Objects are assigned by reference. The object from `Rabbit.prototype` is not duplicated, it's still a single object is referenced both by `Rabbit.prototype` and by the `[[Prototype]]` of `rabbit`. + Objects are assigned by reference. The object from `Rabbit.prototype` is not duplicated, it's still a single object referenced both by `Rabbit.prototype` and by the `[[Prototype]]` of `rabbit`. So when we change its content through one reference, it is visible through the other one. diff --git a/1-js/08-prototypes/02-function-prototype/1-changing-prototype/task.md b/1-js/08-prototypes/02-function-prototype/1-changing-prototype/task.md index 4b8522d3d..2838c125a 100644 --- a/1-js/08-prototypes/02-function-prototype/1-changing-prototype/task.md +++ b/1-js/08-prototypes/02-function-prototype/1-changing-prototype/task.md @@ -20,7 +20,7 @@ alert( rabbit.eats ); // true ``` -1. We added one more string (emphasized), what `alert` shows now? +1. We added one more string (emphasized). What will `alert` show now? ```js function Rabbit() {} @@ -54,7 +54,7 @@ alert( rabbit.eats ); // true alert( rabbit.eats ); // ? ``` -3. Like this (replaced one line)? +3. And like this (replaced one line)? ```js function Rabbit() {} diff --git a/1-js/08-prototypes/02-function-prototype/4-new-object-same-constructor/solution.md b/1-js/08-prototypes/02-function-prototype/4-new-object-same-constructor/solution.md index 43190e163..372d50dd6 100644 --- a/1-js/08-prototypes/02-function-prototype/4-new-object-same-constructor/solution.md +++ b/1-js/08-prototypes/02-function-prototype/4-new-object-same-constructor/solution.md @@ -15,7 +15,7 @@ alert( user2.name ); // Pete (worked!) It worked, because `User.prototype.constructor == User`. -..But if someone, so to say, overwrites `User.prototype` and forgets to recreate `"constructor"`, then it would fail. +..But if someone, so to speak, overwrites `User.prototype` and forgets to recreate `constructor` to reference `User`, then it would fail. For instance: @@ -38,7 +38,12 @@ Why `user2.name` is `undefined`? Here's how `new user.constructor('Pete')` works: 1. First, it looks for `constructor` in `user`. Nothing. -2. Then it follows the prototype chain. The prototype of `user` is `User.prototype`, and it also has nothing. -3. The value of `User.prototype` is a plain object `{}`, its prototype is `Object.prototype`. And there is `Object.prototype.constructor == Object`. So it is used. +2. Then it follows the prototype chain. The prototype of `user` is `User.prototype`, and it also has no `constructor` (because we "forgot" to set it right!). +3. Going further up the chain, `User.prototype` is a plain object, its prototype is the built-in `Object.prototype`. +4. Finally, for the built-in `Object.prototype`, there's a built-in `Object.prototype.constructor == Object`. So it is used. -At the end, we have `let user2 = new Object('Pete')`. The built-in `Object` constructor ignores arguments, it always creates an empty object -- that's what we have in `user2` after all. +Finally, at the end, we have `let user2 = new Object('Pete')`. + +Probably, that's not what we want. We'd like to create `new User`, not `new Object`. That's the outcome of the missing `constructor`. + +(Just in case you're curious, the `new Object(...)` call converts its argument to an object. That's a theoretical thing, in practice no one calls `new Object` with a value, and generally we don't use `new Object` to make objects at all). \ No newline at end of file diff --git a/1-js/08-prototypes/02-function-prototype/article.md b/1-js/08-prototypes/02-function-prototype/article.md index 5511c8ee0..b1ef51826 100644 --- a/1-js/08-prototypes/02-function-prototype/article.md +++ b/1-js/08-prototypes/02-function-prototype/article.md @@ -2,7 +2,7 @@ Remember, new objects can be created with a constructor function, like `new F()`. -If `F.prototype` is an object, then `new` operator uses it to set `[[Prototype]]` for the new object. +If `F.prototype` is an object, then the `new` operator uses it to set `[[Prototype]]` for the new object. ```smart JavaScript had prototypal inheritance from the beginning. It was one of the core features of the language. @@ -41,7 +41,7 @@ That's the resulting picture: On the picture, `"prototype"` is a horizontal arrow, meaning a regular property, and `[[Prototype]]` is vertical, meaning the inheritance of `rabbit` from `animal`. ```smart header="`F.prototype` only used at `new F` time" -`F.prototype` property is only used when `new F` is called, it assigns `[[Prototype]]` of the new object. After that, there's no connection between `F.prototype` and the new object. Think of it as a "one-time gift". +`F.prototype` property is only used when `new F` is called, it assigns `[[Prototype]]` of the new object. If, after the creation, `F.prototype` property changes (`F.prototype = <another object>`), then new objects created by `new F` will have another object as `[[Prototype]]`, but already existing objects keep the old one. ``` @@ -158,11 +158,11 @@ Rabbit.prototype = { In this chapter we briefly described the way of setting a `[[Prototype]]` for objects created via a constructor function. Later we'll see more advanced programming patterns that rely on it. -Everything is quite simple, just few notes to make things clear: +Everything is quite simple, just a few notes to make things clear: -- The `F.prototype` property is not the same as `[[Prototype]]`. The only thing `F.prototype` does: it sets `[[Prototype]]` of new objects when `new F()` is called. -- The value of `F.prototype` should be either an object or null: other values won't work. -- The `"prototype"` property only has such a special effect when is set to a constructor function, and invoked with `new`. +- The `F.prototype` property (don't mistake it for `[[Prototype]]`) sets `[[Prototype]]` of new objects when `new F()` is called. +- The value of `F.prototype` should be either an object or `null`: other values won't work. +- The `"prototype"` property only has such a special effect when set on a constructor function, and invoked with `new`. On regular objects the `prototype` is nothing special: ```js diff --git a/1-js/08-prototypes/03-native-prototypes/2-defer-to-prototype-extended/solution.md b/1-js/08-prototypes/03-native-prototypes/2-defer-to-prototype-extended/solution.md index e3651683f..99c358c9b 100644 --- a/1-js/08-prototypes/03-native-prototypes/2-defer-to-prototype-extended/solution.md +++ b/1-js/08-prototypes/03-native-prototypes/2-defer-to-prototype-extended/solution.md @@ -15,3 +15,27 @@ function f(a, b) { f.defer(1000)(1, 2); // shows 3 after 1 sec ``` + +Please note: we use `this` in `f.apply` to make our decoration work for object methods. + +So if the wrapper function is called as an object method, then `this` is passed to the original method `f`. + +```js run +Function.prototype.defer = function(ms) { + let f = this; + return function(...args) { + setTimeout(() => f.apply(this, args), ms); + } +}; + +let user = { + name: "John", + sayHi() { + alert(this.name); + } +} + +user.sayHi = user.sayHi.defer(1000); + +user.sayHi(); +``` diff --git a/1-js/08-prototypes/03-native-prototypes/article.md b/1-js/08-prototypes/03-native-prototypes/article.md index c5e0dbd60..bdfc86dd8 100644 --- a/1-js/08-prototypes/03-native-prototypes/article.md +++ b/1-js/08-prototypes/03-native-prototypes/article.md @@ -2,7 +2,7 @@ The `"prototype"` property is widely used by the core of JavaScript itself. All built-in constructor functions use it. -We'll see how it is for plain objects first, and then for more complex ones. +First we'll look at the details, and then how to use it for adding new capabilities to built-in objects. ## Object.prototype @@ -33,10 +33,12 @@ We can check it like this: let obj = {}; alert(obj.__proto__ === Object.prototype); // true -// obj.toString === obj.__proto__.toString == Object.prototype.toString + +alert(obj.toString === obj.__proto__.toString); //true +alert(obj.toString === Object.prototype.toString); //true ``` -Please note that there is no additional `[[Prototype]]` in the chain above `Object.prototype`: +Please note that there is no more `[[Prototype]]` in the chain above `Object.prototype`: ```js run alert(Object.prototype.__proto__); // null @@ -46,9 +48,9 @@ alert(Object.prototype.__proto__); // null Other built-in objects such as `Array`, `Date`, `Function` and others also keep methods in prototypes. -For instance, when we create an array `[1, 2, 3]`, the default `new Array()` constructor is used internally. So the array data is written into the new object, and `Array.prototype` becomes its prototype and provides methods. That's very memory-efficient. +For instance, when we create an array `[1, 2, 3]`, the default `new Array()` constructor is used internally. So `Array.prototype` becomes its prototype and provides methods. That's very memory-efficient. -By specification, all of the built-in prototypes have `Object.prototype` on the top. Sometimes people say that "everything inherits from objects". +By specification, all of the built-in prototypes have `Object.prototype` on the top. That's why some people say that "everything inherits from objects". Here's the overall picture (for 3 built-ins to fit): @@ -99,12 +101,12 @@ alert(f.__proto__.__proto__ == Object.prototype); // true, inherit from objects The most intricate thing happens with strings, numbers and booleans. -As we remember, they are not objects. But if we try to access their properties, then temporary wrapper objects are created using built-in constructors `String`, `Number`, `Boolean`, they provide the methods and disappear. +As we remember, they are not objects. But if we try to access their properties, temporary wrapper objects are created using built-in constructors `String`, `Number` and `Boolean`. They provide the methods and disappear. These objects are created invisibly to us and most engines optimize them out, but the specification describes it exactly this way. Methods of these objects also reside in prototypes, available as `String.prototype`, `Number.prototype` and `Boolean.prototype`. ```warn header="Values `null` and `undefined` have no object wrappers" -Special values `null` and `undefined` stand apart. They have no object wrappers, so methods and properties are not available for them. And there are no corresponding prototypes too. +Special values `null` and `undefined` stand apart. They have no object wrappers, so methods and properties are not available for them. And there are no corresponding prototypes either. ``` ## Changing native prototypes [#native-prototype-change] @@ -122,16 +124,16 @@ String.prototype.show = function() { During the process of development, we may have ideas for new built-in methods we'd like to have, and we may be tempted to add them to native prototypes. But that is generally a bad idea. ```warn -Prototypes are global, so it's easy to get a conflict. If two libraries add a method `String.prototype.show`, then one of them will be overwriting the other. +Prototypes are global, so it's easy to get a conflict. If two libraries add a method `String.prototype.show`, then one of them will be overwriting the method of the other. So, generally, modifying a native prototype is considered a bad idea. ``` **In modern programming, there is only one case where modifying native prototypes is approved. That's polyfilling.** -Polyfilling is a term for making a substitute for a method that exists in JavaScript specification, but not yet supported by current JavaScript engine. +Polyfilling is a term for making a substitute for a method that exists in the JavaScript specification, but is not yet supported by a particular JavaScript engine. -Then we may implement it manually and populate the built-in prototype with it. +We may then implement it manually and populate the built-in prototype with it. For instance: @@ -161,7 +163,7 @@ That's when we take a method from one object and copy it into another. Some methods of native prototypes are often borrowed. -For instance, if we're making an array-like object, we may want to copy some array methods to it. +For instance, if we're making an array-like object, we may want to copy some `Array` methods to it. E.g. @@ -179,18 +181,18 @@ obj.join = Array.prototype.join; alert( obj.join(',') ); // Hello,world! ``` -It works, because the internal algorithm of the built-in `join` method only cares about the correct indexes and the `length` property, it doesn't check that the object is indeed the array. And many built-in methods are like that. +It works because the internal algorithm of the built-in `join` method only cares about the correct indexes and the `length` property. It doesn't check if the object is indeed an array. Many built-in methods are like that. Another possibility is to inherit by setting `obj.__proto__` to `Array.prototype`, so all `Array` methods are automatically available in `obj`. But that's impossible if `obj` already inherits from another object. Remember, we only can inherit from one object at a time. -Borrowing methods is flexible, it allows to mix functionality from different objects if needed. +Borrowing methods is flexible, it allows to mix functionalities from different objects if needed. ## Summary - All built-in objects follow the same pattern: - - The methods are stored in the prototype (`Array.prototype`, `Object.prototype`, `Date.prototype` etc). - - The object itself stores only the data (array items, object properties, the date). -- Primitives also store methods in prototypes of wrapper objects: `Number.prototype`, `String.prototype`, `Boolean.prototype`. Only `undefined` and `null` do not have wrapper objects. -- Built-in prototypes can be modified or populated with new methods. But it's not recommended to change them. Probably the only allowable cause is when we add-in a new standard, but not yet supported by the engine JavaScript method. + - The methods are stored in the prototype (`Array.prototype`, `Object.prototype`, `Date.prototype`, etc.) + - The object itself stores only the data (array items, object properties, the date) +- Primitives also store methods in prototypes of wrapper objects: `Number.prototype`, `String.prototype` and `Boolean.prototype`. Only `undefined` and `null` do not have wrapper objects +- Built-in prototypes can be modified or populated with new methods. But it's not recommended to change them. The only allowable case is probably when we add-in a new standard, but it's not yet supported by the JavaScript engine diff --git a/1-js/08-prototypes/04-prototype-methods/2-dictionary-tostring/solution.md b/1-js/08-prototypes/04-prototype-methods/2-dictionary-tostring/solution.md index a92e17900..f3c9cf0e5 100644 --- a/1-js/08-prototypes/04-prototype-methods/2-dictionary-tostring/solution.md +++ b/1-js/08-prototypes/04-prototype-methods/2-dictionary-tostring/solution.md @@ -28,4 +28,4 @@ alert(dictionary); // "apple,__proto__" When we create a property using a descriptor, its flags are `false` by default. So in the code above, `dictionary.toString` is non-enumerable. -See the the chapter [](info:property-descriptors) for review. +See the chapter [](info:property-descriptors) for review. diff --git a/1-js/08-prototypes/04-prototype-methods/article.md b/1-js/08-prototypes/04-prototype-methods/article.md index a0bf20735..e2e9e9416 100644 --- a/1-js/08-prototypes/04-prototype-methods/article.md +++ b/1-js/08-prototypes/04-prototype-methods/article.md @@ -3,15 +3,18 @@ In the first chapter of this section, we mentioned that there are modern methods to setup a prototype. -The `__proto__` is considered outdated and somewhat deprecated (in browser-only part of the JavaScript standard). +Setting or reading the prototype with `obj.__proto__` is considered outdated and somewhat deprecated (moved to the so-called "Annex B" of the JavaScript standard, meant for browsers only). -The modern methods are: +The modern methods to get/set a prototype are: -- [Object.create(proto[, descriptors])](mdn:js/Object/create) -- creates an empty object with given `proto` as `[[Prototype]]` and optional property descriptors. - [Object.getPrototypeOf(obj)](mdn:js/Object/getPrototypeOf) -- returns the `[[Prototype]]` of `obj`. - [Object.setPrototypeOf(obj, proto)](mdn:js/Object/setPrototypeOf) -- sets the `[[Prototype]]` of `obj` to `proto`. -These should be used instead of `__proto__`. +The only usage of `__proto__`, that's not frowned upon, is as a property when creating a new object: `{ __proto__: ... }`. + +Although, there's a special method for this too: + +- [Object.create(proto[, descriptors])](mdn:js/Object/create) -- creates an empty object with given `proto` as `[[Prototype]]` and optional property descriptors. For instance: @@ -22,12 +25,13 @@ let animal = { // create a new object with animal as a prototype *!* -let rabbit = Object.create(animal); +let rabbit = Object.create(animal); // same as {__proto__: animal} */!* alert(rabbit.eats); // true + *!* -alert(Object.getPrototypeOf(rabbit) === animal); // get the prototype of rabbit +alert(Object.getPrototypeOf(rabbit) === animal); // true */!* *!* @@ -35,7 +39,9 @@ Object.setPrototypeOf(rabbit, {}); // change the prototype of rabbit to {} */!* ``` -`Object.create` has an optional second argument: property descriptors. We can provide additional properties to the new object there, like this: +The `Object.create` method is a bit more powerful, as it has an optional second argument: property descriptors. + +We can provide additional properties to the new object there, like this: ```js run let animal = { @@ -56,35 +62,42 @@ The descriptors are in the same format as described in the chapter <info:propert We can use `Object.create` to perform an object cloning more powerful than copying properties in `for..in`: ```js -// fully identical shallow clone of obj -let clone = Object.create(Object.getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj)); +let clone = Object.create( + Object.getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj) +); ``` This call makes a truly exact copy of `obj`, including all properties: enumerable and non-enumerable, data properties and setters/getters -- everything, and with the right `[[Prototype]]`. -## Brief history -If we count all the ways to manage `[[Prototype]]`, there's a lot! Many ways to do the same! +## Brief history -Why so? +There're so many ways to manage `[[Prototype]]`. How did that happen? Why? That's for historical reasons. -- The `"prototype"` property of a constructor function works since very ancient times. -- Later in the year 2012: `Object.create` appeared in the standard. It allowed to create objects with the given prototype, but did not allow to get/set it. So browsers implemented non-standard `__proto__` accessor that allowed to get/set a prototype at any time. -- Later in the year 2015: `Object.setPrototypeOf` and `Object.getPrototypeOf` were added to the standard. The `__proto__` was de-facto implemented everywhere, so it made its way to the Annex B of the standard, that is optional for non-browser environments. +The prototypal inheritance was in the language since its dawn, but the ways to manage it evolved over time. + +- The `prototype` property of a constructor function has worked since very ancient times. It's the oldest way to create objects with a given prototype. +- Later, in the year 2012, `Object.create` appeared in the standard. It gave the ability to create objects with a given prototype, but did not provide the ability to get/set it. Some browsers implemented the non-standard `__proto__` accessor that allowed the user to get/set a prototype at any time, to give more flexibility to developers. +- Later, in the year 2015, `Object.setPrototypeOf` and `Object.getPrototypeOf` were added to the standard, to perform the same functionality as `__proto__`. As `__proto__` was de-facto implemented everywhere, it was kind-of deprecated and made its way to the Annex B of the standard, that is: optional for non-browser environments. +- Later, in the year 2022, it was officially allowed to use `__proto__` in object literals `{...}` (moved out of Annex B), but not as a getter/setter `obj.__proto__` (still in Annex B). + +Why was `__proto__` replaced by the functions `getPrototypeOf/setPrototypeOf`? + +Why was `__proto__` partially rehabilitated and its usage allowed in `{...}`, but not as a getter/setter? -As of now we have all these ways at our disposal. +That's an interesting question, requiring us to understand why `__proto__` is bad. -Why was `__proto__` replaced by the functions? That's an interesting question, requiring us to understand why `__proto__` is bad. Read on to get the answer. +And soon we'll get the answer. -```warn header="Don't reset `[[Prototype]]` unless the speed doesn't matter" -Technically, we can get/set `[[Prototype]]` at any time. But usually we only set it once at the object creation time, and then do not modify: `rabbit` inherits from `animal`, and that is not going to change. +```warn header="Don't change `[[Prototype]]` on existing objects if speed matters" +Technically, we can get/set `[[Prototype]]` at any time. But usually we only set it once at the object creation time and don't modify it anymore: `rabbit` inherits from `animal`, and that is not going to change. -And JavaScript engines are highly optimized to that. Changing a prototype "on-the-fly" with `Object.setPrototypeOf` or `obj.__proto__=` is a very slow operation, it breaks internal optimizations for object property access operations. So evade it unless you know what you're doing, or JavaScript speed totally doesn't matter for you. +And JavaScript engines are highly optimized for this. Changing a prototype "on-the-fly" with `Object.setPrototypeOf` or `obj.__proto__=` is a very slow operation as it breaks internal optimizations for object property access operations. So avoid it unless you know what you're doing, or JavaScript speed totally doesn't matter for you. ``` -## "Very plain" objects +## "Very plain" objects [#very-plain] As we know, objects can be used as associative arrays to store key/value pairs. @@ -101,37 +114,52 @@ obj[key] = "some value"; alert(obj[key]); // [object Object], not "some value"! ``` -Here if the user types in `__proto__`, the assignment is ignored! +Here, if the user types in `__proto__`, the assignment in line 4 is ignored! -That shouldn't surprise us. The `__proto__` property is special: it must be either an object or `null`, a string can not become a prototype. +That could surely be surprising for a non-developer, but pretty understandable for us. The `__proto__` property is special: it must be either an object or `null`. A string can not become a prototype. That's why assigning a string to `__proto__` is ignored. But we didn't *intend* to implement such behavior, right? We want to store key/value pairs, and the key named `"__proto__"` was not properly saved. So that's a bug! -Here the consequences are not terrible. But in other cases the prototype may indeed be changed, so the execution may go wrong in totally unexpected ways. +Here the consequences are not terrible. But in other cases we may be storing objects instead of strings in `obj`, and then the prototype will indeed be changed. As a result, the execution will go wrong in totally unexpected ways. -What's worst -- usually developers do not think about such possibility at all. That makes such bugs hard to notice and even turn them into vulnerabilities, especially when JavaScript is used on server-side. +What's worse -- usually developers do not think about such possibility at all. That makes such bugs hard to notice and even turn them into vulnerabilities, especially when JavaScript is used on server-side. -Unexpected things also may happen when accessing `toString` property -- that's a function by default, and other built-in properties. +Unexpected things also may happen when assigning to `obj.toString`, as it's a built-in object method. -How to evade the problem? +How can we avoid this problem? -First, we can just switch to using `Map`, then everything's fine. +First, we can just switch to using `Map` for storage instead of plain objects, then everything's fine: -But `Object` also can serve us well here, because language creators gave a thought to that problem long ago. +```js run +let map = new Map(); + +let key = prompt("What's the key?", "__proto__"); +map.set(key, "some value"); + +<<<<<<< HEAD +======= +alert(map.get(key)); // "some value" (as intended) +``` -The `__proto__` is not a property of an object, but an accessor property of `Object.prototype`: +...But `Object` syntax is often more appealing, as it's more concise. +Fortunately, we *can* use objects, because language creators gave thought to that problem long ago. + +As we know, `__proto__` is not a property of an object, but an accessor property of `Object.prototype`: + +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  So, if `obj.__proto__` is read or set, the corresponding getter/setter is called from its prototype, and it gets/sets `[[Prototype]]`. As it was said in the beginning of this tutorial section: `__proto__` is a way to access `[[Prototype]]`, it is not `[[Prototype]]` itself. -Now, if we want to use an object as an associative array, we can do it with a little trick: +Now, if we intend to use an object as an associative array and be free of such problems, we can do it with a little trick: ```js run *!* let obj = Object.create(null); +// or: obj = { __proto__: null } */!* let key = prompt("What's the key?", "__proto__"); @@ -146,7 +174,7 @@ alert(obj[key]); // "some value" So, there is no inherited getter/setter for `__proto__`. Now it is processed as a regular data property, so the example above works right. -We can call such object "very plain" or "pure dictionary objects", because they are even simpler than regular plain object `{...}`. +We can call such objects "very plain" or "pure dictionary" objects, because they are even simpler than the regular plain object `{...}`. A downside is that such objects lack any built-in object methods, e.g. `toString`: @@ -160,7 +188,7 @@ alert(obj); // Error (no toString) ...But that's usually fine for associative arrays. -Please note that most object-related methods are `Object.something(...)`, like `Object.keys(obj)` -- they are not in the prototype, so they will keep working on such objects: +Note that most object-related methods are `Object.something(...)`, like `Object.keys(obj)` -- they are not in the prototype, so they will keep working on such objects: ```js run @@ -173,31 +201,26 @@ alert(Object.keys(chineseDictionary)); // hello,bye ## Summary -Modern methods to setup and directly access the prototype are: - -- [Object.create(proto[, descriptors])](mdn:js/Object/create) -- creates an empty object with given `proto` as `[[Prototype]]` (can be `null`) and optional property descriptors. -- [Object.getPrototypeOf(obj)](mdn:js/Object.getPrototypeOf) -- returns the `[[Prototype]]` of `obj` (same as `__proto__` getter). -- [Object.setPrototypeOf(obj, proto)](mdn:js/Object.setPrototypeOf) -- sets the `[[Prototype]]` of `obj` to `proto` (same as `__proto__` setter). +- To create an object with the given prototype, use: -The built-in `__proto__` getter/setter is unsafe if we'd want to put user-generated keys in to an object. Just because a user may enter "__proto__" as the key, and there'll be an error with hopefully easy, but generally unpredictable consequences. + - literal syntax: `{ __proto__: ... }`, allows to specify multiple properties + - or [Object.create(proto[, descriptors])](mdn:js/Object/create), allows to specify property descriptors. -So we can either use `Object.create(null)` to create a "very plain" object without `__proto__`, or stick to `Map` objects for that. + The `Object.create` provides an easy way to shallow-copy an object with all descriptors: -Also, `Object.create` provides an easy way to shallow-copy an object with all descriptors: + ```js + let clone = Object.create(Object.getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj)); + ``` -```js -let clone = Object.create(Object.getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj)); -``` +- Modern methods to get/set the prototype are: + - [Object.getPrototypeOf(obj)](mdn:js/Object/getPrototypeOf) -- returns the `[[Prototype]]` of `obj` (same as `__proto__` getter). + - [Object.setPrototypeOf(obj, proto)](mdn:js/Object/setPrototypeOf) -- sets the `[[Prototype]]` of `obj` to `proto` (same as `__proto__` setter). -- [Object.keys(obj)](mdn:js/Object/keys) / [Object.values(obj)](mdn:js/Object/values) / [Object.entries(obj)](mdn:js/Object/entries) -- returns an array of enumerable own string property names/values/key-value pairs. -- [Object.getOwnPropertySymbols(obj)](mdn:js/Object/getOwnPropertySymbols) -- returns an array of all own symbolic property names. -- [Object.getOwnPropertyNames(obj)](mdn:js/Object/getOwnPropertyNames) -- returns an array of all own string property names. -- [Reflect.ownKeys(obj)](mdn:js/Reflect/ownKeys) -- returns an array of all own property names. -- [obj.hasOwnProperty(key)](mdn:js/Object/hasOwnProperty): it returns `true` if `obj` has its own (not inherited) property named `key`. +- Getting/setting the prototype using the built-in `__proto__` getter/setter isn't recommended, it's now in the Annex B of the specification. -We also made it clear that `__proto__` is a getter/setter for `[[Prototype]]` and resides in `Object.prototype`, just as other methods. +- We also covered prototype-less objects, created with `Object.create(null)` or `{__proto__: null}`. -We can create an object without a prototype by `Object.create(null)`. Such objects are used as "pure dictionaries", they have no issues with `"__proto__"` as the key. + These objects are used as dictionaries, to store any (possibly user-generated) keys. -All methods that return object properties (like `Object.keys` and others) -- return "own" properties. If we want inherited ones, then we can use `for..in`. + Normally, objects inherit built-in methods and `__proto__` getter/setter from `Object.prototype`, making corresponding keys "occupied" and potentially causing side effects. With `null` prototype, objects are truly empty. diff --git a/1-js/09-classes/01-class/1-rewrite-to-class/task.md b/1-js/09-classes/01-class/1-rewrite-to-class/task.md index 05365e410..4477de679 100644 --- a/1-js/09-classes/01-class/1-rewrite-to-class/task.md +++ b/1-js/09-classes/01-class/1-rewrite-to-class/task.md @@ -4,6 +4,6 @@ importance: 5 # Rewrite to class -The `Clock` class is written in functional style. Rewrite it the "class" syntax. +The `Clock` class (see the sandbox) is written in functional style. Rewrite it in the "class" syntax. P.S. The clock ticks in the console, open it to see. diff --git a/1-js/09-classes/01-class/article.md b/1-js/09-classes/01-class/article.md index b8bd4f339..c33873af6 100644 --- a/1-js/09-classes/01-class/article.md +++ b/1-js/09-classes/01-class/article.md @@ -25,7 +25,7 @@ class MyClass { } ``` -Then `new MyClass()` creates a new object with all the listed methods. +Then use `new MyClass()` to create a new object with all the listed methods. The `constructor()` method is called automatically by `new`, so we can initialize the object there. @@ -51,9 +51,9 @@ user.sayHi(); When `new User("John")` is called: 1. A new object is created. -2. The `constructor` runs with the given argument and assigns `this.name` to it. +2. The `constructor` runs with the given argument and assigns it to `this.name`. -...Then we can call methods, such as `user.sayHi`. +...Then we can call object methods, such as `user.sayHi()`. ```warn header="No comma between class methods" @@ -64,11 +64,11 @@ The notation here is not to be confused with object literals. Within the class, ## What is a class? -So, what exactly is a `class`? That's not an entirely new language-level entity, as one might think. +So, what exactly is a `class`? That's not an entirely new language-level entity, as one might think. Let's unveil any magic and see what a class really is. That'll help in understanding many complex aspects. -In JavaScript, a class is a kind of a function. +In JavaScript, a class is a kind of function. Here, take a look: @@ -85,19 +85,21 @@ alert(typeof User); // function ``` What `class User {...}` construct really does is: -1. Creates a function named `User`, that becomes the result of the class declaration. - - The function code is taken from the `constructor` method (assumed empty if we don't write such method). -3. Stores all methods, such as `sayHi`, in `User.prototype`. -Afterwards, for new objects, when we call a method, it's taken from the prototype, just as described in the chapter <info:function-prototype>. So `new User` object has access to class methods. +1. Creates a function named `User`, that becomes the result of the class declaration. The function code is taken from the `constructor` method (assumed empty if we don't write such method). +2. Stores class methods, such as `sayHi`, in `User.prototype`. -We can illustrate the result of `class User` as: +After `new User` object is created, when we call its method, it's taken from the prototype, just as described in the chapter <info:function-prototype>. So the object has access to class methods. +<<<<<<< HEAD +======= +We can illustrate the result of `class User` declaration as: + +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  Here's the code to introspect it: - ```js run class User { constructor(name) { this.name = name; } @@ -111,15 +113,15 @@ alert(typeof User); // function alert(User === User.prototype.constructor); // true // The methods are in User.prototype, e.g: -alert(User.prototype.sayHi); // alert(this.name); +alert(User.prototype.sayHi); // the code of the sayHi method // there are exactly two methods in the prototype alert(Object.getOwnPropertyNames(User.prototype)); // constructor, sayHi ``` -## Not just a syntax sugar +## Not just a syntactic sugar -Sometimes people say that `class` is a "syntax sugar" in JavaScript, because we could actually declare the same without `class` keyword at all: +Sometimes people say that `class` is a "syntactic sugar" (syntax that is designed to make things easier to read, but doesn't introduce anything new), because we could actually declare the same thing without using the `class` keyword at all: ```js run // rewriting class User in pure functions @@ -128,7 +130,7 @@ Sometimes people say that `class` is a "syntax sugar" in JavaScript, because we function User(name) { this.name = name; } -// any function prototype has constructor property by default, +// a function prototype has "constructor" property by default, // so we don't need to create it // 2. Add the method to prototype @@ -141,13 +143,13 @@ let user = new User("John"); user.sayHi(); ``` -The result of this definition is about the same. So, there are indeed reasons why `class` can be considered a syntax sugar to define a constructor together with its prototype methods. +The result of this definition is about the same. So, there are indeed reasons why `class` can be considered a syntactic sugar to define a constructor together with its prototype methods. -Although, there are important differences. +Still, there are important differences. -1. First, a function created by `class` is labelled by a special internal property `[[FunctionKind]]:"classConstructor"`. So it's not entirely the same as creating it manually. +1. First, a function created by `class` is labelled by a special internal property `[[IsClassConstructor]]: true`. So it's not entirely the same as creating it manually. - Unlike a regular function, a class constructor can't be called without `new`: + The language checks for that property in a variety of places. For example, unlike a regular function, it must be called with `new`: ```js run class User { @@ -167,21 +169,21 @@ Although, there are important differences. alert(User); // class User { ... } ``` + There are other differences, we'll see them soon. -2. Class methods are non-enumerable +2. Class methods are non-enumerable. A class definition sets `enumerable` flag to `false` for all methods in the `"prototype"`. That's good, because if we `for..in` over an object, we usually don't want its class methods. -3. Classes always `use strict` +3. Classes always `use strict`. All code inside the class construct is automatically in strict mode. - -Also, in addition to its basic operation, the `class` syntax brings many other features with it which we'll explore later. +Besides, `class` syntax brings many other features that we'll explore later. ## Class Expression -Just like functions, classes can be defined inside another expression, passed around, returned, assigned etc. +Just like functions, classes can be defined inside another expression, passed around, returned, assigned, etc. Here's an example of a class expression: @@ -193,24 +195,24 @@ let User = class { }; ``` -Similar to Named Function Expressions, class expressions may or may not have a name. +Similar to Named Function Expressions, class expressions may have a name. If a class expression has a name, it's visible inside the class only: ```js run -// "Named Class Expression" (alas, no such term, but that's what's going on) +// "Named Class Expression" +// (no such term in the spec, but that's similar to Named Function Expression) let User = class *!*MyClass*/!* { sayHi() { - alert(MyClass); // MyClass is visible only inside the class + alert(MyClass); // MyClass name is visible only inside the class } }; new User().sayHi(); // works, shows MyClass definition -alert(MyClass); // error, MyClass not visible outside of the class +alert(MyClass); // error, MyClass name isn't visible outside of the class ``` - We can even make classes dynamically "on-demand", like this: ```js run @@ -219,7 +221,7 @@ function makeClass(phrase) { return class { sayHi() { alert(phrase); - }; + } }; } @@ -230,9 +232,9 @@ new User().sayHi(); // Hello ``` -## Getters/setters, other shorthands +## Getters/setters -Classes also include getters/setters, generators, computed properties etc. +Just like literal objects, classes may include getters/setters, computed properties etc. Here's an example for `user.name` implemented using `get/set`: @@ -241,7 +243,7 @@ class User { constructor(name) { // invokes the setter - this._name = name; + this.name = name; } *!* @@ -265,31 +267,21 @@ class User { let user = new User("John"); alert(user.name); // John -user = new User(""); // Name too short. +user = new User(""); // Name is too short. ``` -Internally, getters and setters are created on `User.prototype`, like this: +Technically, such class declaration works by creating getters and setters in `User.prototype`. -```js -Object.defineProperties(User.prototype, { - name: { - get() { - return this._name - }, - set(name) { - // ... - } - } -}); -``` +## Computed names [...] -Here's an example with computed properties: +Here's an example with a computed method name using brackets `[...]`: ```js run -function f() { return "sayHi"; } - class User { - [f()]() { + +*!* + ['say' + 'Hi']() { +*/!* alert("Hello"); } @@ -298,56 +290,127 @@ class User { new User().sayHi(); ``` -For a generator method, similarly, prepend it with `*`. +Such features are easy to remember, as they resemble that of literal objects. -## Class properties +## Class fields ```warn header="Old browsers may need a polyfill" -Class-level properties are a recent addition to the language. +Class fields are a recent addition to the language. ``` -In the example above, `User` only had methods. Let's add a property: +Previously, our classes only had methods. + +"Class fields" is a syntax that allows to add any properties. + +For instance, let's add `name` property to `class User`: ```js run class User { - name = "Anonymous"; +*!* + name = "John"; +*/!* sayHi() { alert(`Hello, ${this.name}!`); } } -new User().sayHi(); +new User().sayHi(); // Hello, John! ``` -The property is not placed into `User.prototype`. Instead, it is created by `new`, separately for every object. So, the property will never be shared between different objects of the same class. +So, we just write "<property name> = <value>" in the declaration, and that's it. +The important difference of class fields is that they are set on individual objects, not `User.prototype`: -## Summary +```js run +class User { +*!* + name = "John"; +*/!* +} -JavaScript provides many ways to create a class. +let user = new User(); +alert(user.name); // John +alert(User.prototype.name); // undefined +``` -First, as per the general object-oriented terminology, a class is something that provides "object templates", allows to create same-structured objects. +We can also assign values using more complex expressions and function calls: -When we say "a class", that doesn't necessary means the `class` keyword. +```js run +class User { +*!* + name = prompt("Name, please?", "John"); +*/!* +} -This is a class: +let user = new User(); +alert(user.name); // John +``` -```js -function User(name) { - this.sayHi = function() { - alert(name); + +### Making bound methods with class fields + +As demonstrated in the chapter <info:bind> functions in JavaScript have a dynamic `this`. It depends on the context of the call. + +So if an object method is passed around and called in another context, `this` won't be a reference to its object any more. + +For instance, this code will show `undefined`: + +```js run +class Button { + constructor(value) { + this.value = value; + } + + click() { + alert(this.value); + } +} + +let button = new Button("hello"); + +*!* +setTimeout(button.click, 1000); // undefined +*/!* +``` + +The problem is called "losing `this`". + +There are two approaches to fixing it, as discussed in the chapter <info:bind>: + +1. Pass a wrapper-function, such as `setTimeout(() => button.click(), 1000)`. +2. Bind the method to object, e.g. in the constructor. + +Class fields provide another, quite elegant syntax: + +```js run +class Button { + constructor(value) { + this.value = value; } +*!* + click = () => { + alert(this.value); + } +*/!* } + +let button = new Button("hello"); + +setTimeout(button.click, 1000); // hello ``` -...But in most cases `class` keyword is used, as it provides great syntax and many additional features. +The class field `click = () => {...}` is created on a per-object basis, there's a separate function for each `Button` object, with `this` inside it referencing that object. We can pass `button.click` around anywhere, and the value of `this` will always be correct. + +That's especially useful in browser environment, for event listeners. + +## Summary The basic class syntax looks like this: ```js class MyClass { - prop = value; // field + prop = value; // property constructor(...) { // constructor // ... @@ -358,11 +421,11 @@ class MyClass { get something(...) {} // getter method set something(...) {} // setter method - [Symbol.iterator]() {} // method with computed name/symbol name + [Symbol.iterator]() {} // method with computed name (symbol here) // ... } ``` -`MyClass` is technically a function, while methods are written to `MyClass.prototype`. +`MyClass` is technically a function (the one that we provide as `constructor`), while methods, getters and setters are written to `MyClass.prototype`. In the next chapters we'll learn more about classes, including inheritance and other features. diff --git a/1-js/09-classes/02-class-inheritance/2-clock-class-extended/solution.view/extended-clock.js b/1-js/09-classes/02-class-inheritance/2-clock-class-extended/solution.view/extended-clock.js index ca613ca5e..be2053cfc 100644 --- a/1-js/09-classes/02-class-inheritance/2-clock-class-extended/solution.view/extended-clock.js +++ b/1-js/09-classes/02-class-inheritance/2-clock-class-extended/solution.view/extended-clock.js @@ -1,7 +1,7 @@ class ExtendedClock extends Clock { constructor(options) { super(options); - let { precision=1000 } = options; + let { precision = 1000 } = options; this.precision = precision; } diff --git a/1-js/09-classes/02-class-inheritance/article.md b/1-js/09-classes/02-class-inheritance/article.md index 4b281bafe..b917c3bb9 100644 --- a/1-js/09-classes/02-class-inheritance/article.md +++ b/1-js/09-classes/02-class-inheritance/article.md @@ -1,9 +1,13 @@ # Class inheritance -Let's say we have two classes. +Class inheritance is a way for one class to extend another class. -`Animal`: +So we can create new functionality on top of the existing. + +## The "extends" keyword + +Let's say we have class `Animal`: ```js class Animal { @@ -12,24 +16,36 @@ class Animal { this.name = name; } run(speed) { - this.speed += speed; + this.speed = speed; alert(`${this.name} runs with speed ${this.speed}.`); } stop() { this.speed = 0; - alert(`${this.name} stopped.`); + alert(`${this.name} stands still.`); } } let animal = new Animal("My animal"); ``` +<<<<<<< HEAD  +======= +Here's how we can represent `animal` object and `Animal` class graphically: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b + + + +...And we would like to create another `class Rabbit`. +As rabbits are animals, `Rabbit` class should be based on `Animal`, have access to animal methods, so that rabbits can do what "generic" animals can do. -...And `Rabbit`: +The syntax to extend another class is: `class Child extends Parent`. + +Let's create `class Rabbit` that inherits from `Animal`: ```js +<<<<<<< HEAD class Rabbit { constructor(name) { this.name = name; @@ -70,6 +86,8 @@ class Animal { } // Inherit from Animal by specifying "extends Animal" +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b *!* class Rabbit extends Animal { */!* @@ -84,15 +102,18 @@ rabbit.run(5); // White Rabbit runs with speed 5. rabbit.hide(); // White Rabbit hides! ``` -Now the `Rabbit` code became a bit shorter, as it uses `Animal` constructor by default, and it also can `run`, as animals do. +Object of `Rabbit` class have access both to `Rabbit` methods, such as `rabbit.hide()`, and also to `Animal` methods, such as `rabbit.run()`. -Internally, `extends` keyword adds `[[Prototype]]` reference from `Rabbit.prototype` to `Animal.prototype`: +Internally, `extends` keyword works using the good old prototype mechanics. It sets `Rabbit.prototype.[[Prototype]]` to `Animal.prototype`. So, if a method is not found in `Rabbit.prototype`, JavaScript takes it from `Animal.prototype`.  -So, if a method is not found in `Rabbit.prototype`, JavaScript takes it from `Animal.prototype`. +For instance, to find `rabbit.run` method, the engine checks (bottom-up on the picture): +1. The `rabbit` object (has no `run`). +2. Its prototype, that is `Rabbit.prototype` (has `hide`, but not `run`). +3. Its prototype, that is (due to `extends`) `Animal.prototype`, that finally has the `run` method. -As we can recall from the chapter <info:native-prototypes>, JavaScript uses the same prototypal inheritance for build-in objects. E.g. `Date.prototype.[[Prototype]]` is `Object.prototype`, so dates have generic object methods. +As we can recall from the chapter <info:native-prototypes>, JavaScript itself uses prototypal inheritance for built-in objects. E.g. `Date.prototype.[[Prototype]]` is `Object.prototype`. That's why dates have access to generic object methods. ````smart header="Any expression is allowed after `extends`" Class syntax allows to specify not just a class, but any expression after `extends`. @@ -102,8 +123,8 @@ For instance, a function call that generates the parent class: ```js run function f(phrase) { return class { - sayHi() { alert(phrase) } - } + sayHi() { alert(phrase); } + }; } *!* @@ -119,20 +140,20 @@ That may be useful for advanced programming patterns when we use functions to ge ## Overriding a method -Now let's move forward and override a method. As of now, `Rabbit` inherits the `stop` method that sets `this.speed = 0` from `Animal`. +Now let's move forward and override a method. By default, all methods that are not specified in `class Rabbit` are taken directly "as is" from `class Animal`. -If we specify our own `stop` in `Rabbit`, then it will be used instead: +But if we specify our own method in `Rabbit`, such as `stop()` then it will be used instead: ```js class Rabbit extends Animal { stop() { - // ...this will be used for rabbit.stop() + // ...now this will be used for rabbit.stop() + // instead of stop() from class Animal } } ``` - -...But usually we don't want to totally replace a parent method, but rather to build on top of it, tweak or extend its functionality. We do something in our method, but call the parent method before/after it or in the process. +Usually, however, we don't want to totally replace a parent method, but rather to build on top of it to tweak or extend its functionality. We do something in our method, but call the parent method before/after it or in the process. Classes provide `"super"` keyword for that. @@ -150,13 +171,13 @@ class Animal { } run(speed) { - this.speed += speed; + this.speed = speed; alert(`${this.name} runs with speed ${this.speed}.`); } stop() { this.speed = 0; - alert(`${this.name} stopped.`); + alert(`${this.name} stands still.`); } } @@ -177,7 +198,7 @@ class Rabbit extends Animal { let rabbit = new Rabbit("White Rabbit"); rabbit.run(5); // White Rabbit runs with speed 5. -rabbit.stop(); // White Rabbit stopped. White rabbit hides! +rabbit.stop(); // White Rabbit stands still. White Rabbit hides! ``` Now `Rabbit` has the `stop` method that calls the parent `super.stop()` in the process. @@ -186,6 +207,7 @@ Now `Rabbit` has the `stop` method that calls the parent `super.stop()` in the p As was mentioned in the chapter <info:arrow-functions>, arrow functions do not have `super`. If accessed, it's taken from the outer function. For instance: + ```js class Rabbit extends Animal { stop() { @@ -202,14 +224,13 @@ setTimeout(function() { super.stop() }, 1000); ``` ```` - ## Overriding constructor With constructors it gets a little bit tricky. -Till now, `Rabbit` did not have its own `constructor`. +Until now, `Rabbit` did not have its own `constructor`. -According to the [specification](https://tc39.github.io/ecma262/#sec-runtime-semantics-classdefinitionevaluation), if a class extends another class and has no `constructor`, then the following `constructor` is generated: +According to the [specification](https://tc39.github.io/ecma262/#sec-runtime-semantics-classdefinitionevaluation), if a class extends another class and has no `constructor`, then the following "empty" `constructor` is generated: ```js class Rabbit extends Animal { @@ -256,22 +277,24 @@ let rabbit = new Rabbit("White Rabbit", 10); // Error: this is not defined. Whoops! We've got an error. Now we can't create rabbits. What went wrong? -The short answer is: constructors in inheriting classes must call `super(...)`, and (!) do it before using `this`. +The short answer is: + +- **Constructors in inheriting classes must call `super(...)`, and (!) do it before using `this`.** ...But why? What's going on here? Indeed, the requirement seems strange. -Of course, there's an explanation. Let's get into details, so you'd really understand what's going on. +Of course, there's an explanation. Let's get into details, so you'll really understand what's going on. -In JavaScript, there's a distinction between a "constructor function of an inheriting class" and all others. In an inheriting class, the corresponding constructor function is labelled with a special internal property `[[ConstructorKind]]:"derived"`. +In JavaScript, there's a distinction between a constructor function of an inheriting class (so-called "derived constructor") and other functions. A derived constructor has a special internal property `[[ConstructorKind]]:"derived"`. That's a special internal label. -The difference is: +That label affects its behavior with `new`. -- When a normal constructor runs, it creates an empty object as `this` and continues with it. -- But when a derived constructor runs, it doesn't do it. It expects the parent constructor to do this job. +- When a regular function is executed with `new`, it creates an empty object and assigns it to `this`. +- But when a derived constructor runs, it doesn't do this. It expects the parent constructor to do this job. -So if we're making a constructor of our own, then we must call `super`, because otherwise the object with `this` reference to it won't be created. And we'll get an error. +So a derived constructor must call `super` in order to execute its parent (base) constructor, otherwise the object for `this` won't be created. And we'll get an error. -For `Rabbit` to work, we need to call `super()` before using `this`, like here: +For the `Rabbit` constructor to work, it needs to call `super()` before using `this`, like here: ```js run class Animal { @@ -304,20 +327,121 @@ alert(rabbit.earLength); // 10 */!* ``` +### Overriding class fields: a tricky note + +```warn header="Advanced note" +This note assumes you have a certain experience with classes, maybe in other programming languages. + +It provides better insight into the language and also explains the behavior that might be a source of bugs (but not very often). + +If you find it difficult to understand, just go on, continue reading, then return to it some time later. +``` + +We can override not only methods, but also class fields. + +Although, there's a tricky behavior when we access an overridden field in parent constructor, quite different from most other programming languages. + +Consider this example: + +```js run +class Animal { + name = 'animal'; + + constructor() { + alert(this.name); // (*) + } +} + +class Rabbit extends Animal { + name = 'rabbit'; +} + +new Animal(); // animal +*!* +new Rabbit(); // animal +*/!* +``` + +Here, class `Rabbit` extends `Animal` and overrides the `name` field with its own value. + +There's no own constructor in `Rabbit`, so `Animal` constructor is called. + +What's interesting is that in both cases: `new Animal()` and `new Rabbit()`, the `alert` in the line `(*)` shows `animal`. + +**In other words, the parent constructor always uses its own field value, not the overridden one.** + +What's odd about it? + +If it's not clear yet, please compare with methods. + +Here's the same code, but instead of `this.name` field we call `this.showName()` method: + +```js run +class Animal { + showName() { // instead of this.name = 'animal' + alert('animal'); + } + + constructor() { + this.showName(); // instead of alert(this.name); + } +} + +class Rabbit extends Animal { + showName() { + alert('rabbit'); + } +} + +new Animal(); // animal +*!* +new Rabbit(); // rabbit +*/!* +``` + +Please note: now the output is different. + +And that's what we naturally expect. When the parent constructor is called in the derived class, it uses the overridden method. + +...But for class fields it's not so. As said, the parent constructor always uses the parent field. + +Why is there a difference? + +Well, the reason is the field initialization order. The class field is initialized: +- Before constructor for the base class (that doesn't extend anything), +- Immediately after `super()` for the derived class. + +In our case, `Rabbit` is the derived class. There's no `constructor()` in it. As said previously, that's the same as if there was an empty constructor with only `super(...args)`. + +So, `new Rabbit()` calls `super()`, thus executing the parent constructor, and (per the rule for derived classes) only after that its class fields are initialized. At the time of the parent constructor execution, there are no `Rabbit` class fields yet, that's why `Animal` fields are used. + +This subtle difference between fields and methods is specific to JavaScript. + +Luckily, this behavior only reveals itself if an overridden field is used in the parent constructor. Then it may be difficult to understand what's going on, so we're explaining it here. + +If it becomes a problem, one can fix it by using methods or getters/setters instead of fields. ## Super: internals, [[HomeObject]] -Let's get a little deeper under the hood of `super`. We'll see some interesting things by the way. +```warn header="Advanced information" +If you're reading the tutorial for the first time - this section may be skipped. -First to say, from all that we've learned till now, it's impossible for `super` to work. +It's about the internal mechanisms behind inheritance and `super`. +``` -Yeah, indeed, let's ask ourselves, how it could technically work? When an object method runs, it gets the current object as `this`. If we call `super.method()` then, how to retrieve the `method`? Naturally, we need to take the `method` from the prototype of the current object. How, technically, we (or a JavaScript engine) can do it? +Let's get a little deeper under the hood of `super`. We'll see some interesting things along the way. -Maybe we can get the method from `[[Prototype]]` of `this`, as `this.__proto__.method`? Unfortunately, that doesn't work. +First to say, from all that we've learned till now, it's impossible for `super` to work at all! -Let's try to do it. Without classes, using plain objects for the sake of simplicity. +Yeah, indeed, let's ask ourselves, how it should technically work? When an object method runs, it gets the current object as `this`. If we call `super.method()` then, the engine needs to get the `method` from the prototype of the current object. But how? -Here, `rabbit.eat()` should call `animal.eat()` method of the parent object: +The task may seem simple, but it isn't. The engine knows the current object `this`, so it could get the parent `method` as `this.__proto__.method`. Unfortunately, such a "naive" solution won't work. + +Let's demonstrate the problem. Without classes, using plain objects for the sake of simplicity. + +You may skip this part and go below to the `[[HomeObject]]` subsection if you don't want to know the details. That won't harm. Or read on if you're interested in understanding things in-depth. + +In the example below, `rabbit.__proto__ = animal`. Now let's try: in `rabbit.eat()` we'll call `animal.eat()`, using `this.__proto__`: ```js run let animal = { @@ -414,18 +538,16 @@ The problem can't be solved by using `this` alone. To provide the solution, JavaScript adds one more special internal property for functions: `[[HomeObject]]`. -**When a function is specified as a class or object method, its `[[HomeObject]]` property becomes that object.** - -This actually violates the idea of "unbound" functions, because methods remember their objects. And `[[HomeObject]]` can't be changed, so this bound is forever. So that's a very important change in the language. +When a function is specified as a class or object method, its `[[HomeObject]]` property becomes that object. -But this change is safe. `[[HomeObject]]` is used only for calling parent methods in `super`, to resolve the prototype. So it doesn't break compatibility. +Then `super` uses it to resolve the parent prototype and its methods. -Let's see how it works for `super` -- again, using plain objects: +Let's see how it works, first with plain objects: ```js run let animal = { name: "Animal", - eat() { // [[HomeObject]] == animal + eat() { // animal.eat.[[HomeObject]] == animal alert(`${this.name} eats.`); } }; @@ -433,7 +555,7 @@ let animal = { let rabbit = { __proto__: animal, name: "Rabbit", - eat() { // [[HomeObject]] == rabbit + eat() { // rabbit.eat.[[HomeObject]] == rabbit super.eat(); } }; @@ -441,25 +563,85 @@ let rabbit = { let longEar = { __proto__: rabbit, name: "Long Ear", - eat() { // [[HomeObject]] == longEar + eat() { // longEar.eat.[[HomeObject]] == longEar super.eat(); } }; *!* +// works correctly longEar.eat(); // Long Ear eats. */!* ``` -Every method remembers its object in the internal `[[HomeObject]]` property. Then `super` uses it to resolve the parent prototype. +It works as intended, due to `[[HomeObject]]` mechanics. A method, such as `longEar.eat`, knows its `[[HomeObject]]` and takes the parent method from its prototype. Without any use of `this`. + +### Methods are not "free" + +As we've known before, generally functions are "free", not bound to objects in JavaScript. So they can be copied between objects and called with another `this`. + +The very existence of `[[HomeObject]]` violates that principle, because methods remember their objects. `[[HomeObject]]` can't be changed, so this bond is forever. -`[[HomeObject]]` is defined for methods defined both in classes and in plain objects. But for objects, methods must be specified exactly the given way: as `method()`, not as `"method: function()"`. +The only place in the language where `[[HomeObject]]` is used -- is `super`. So, if a method does not use `super`, then we can still consider it free and copy between objects. But with `super` things may go wrong. + +Here's the demo of a wrong `super` result after copying: + +```js run +let animal = { + sayHi() { + alert(`I'm an animal`); + } +}; + +// rabbit inherits from animal +let rabbit = { + __proto__: animal, + sayHi() { + super.sayHi(); + } +}; + +let plant = { + sayHi() { + alert("I'm a plant"); + } +}; + +// tree inherits from plant +let tree = { + __proto__: plant, +*!* + sayHi: rabbit.sayHi // (*) +*/!* +}; + +*!* +tree.sayHi(); // I'm an animal (?!?) +*/!* +``` + +A call to `tree.sayHi()` shows "I'm an animal". Definitely wrong. + +The reason is simple: +- In the line `(*)`, the method `tree.sayHi` was copied from `rabbit`. Maybe we just wanted to avoid code duplication? +- Its `[[HomeObject]]` is `rabbit`, as it was created in `rabbit`. There's no way to change `[[HomeObject]]`. +- The code of `tree.sayHi()` has `super.sayHi()` inside. It goes up from `rabbit` and takes the method from `animal`. + +Here's the diagram of what happens: + + + +### Methods, not function properties + +`[[HomeObject]]` is defined for methods both in classes and in plain objects. But for objects, methods must be specified exactly as `method()`, not as `"method: function()"`. + +The difference may be non-essential for us, but it's important for JavaScript. In the example below a non-method syntax is used for comparison. `[[HomeObject]]` property is not set and the inheritance doesn't work: ```js run let animal = { - eat: function() { // should be the short syntax: eat() {...} + eat: function() { // intentionally writing like this instead of eat() {... // ... } }; @@ -475,3 +657,18 @@ let rabbit = { rabbit.eat(); // Error calling super (because there's no [[HomeObject]]) */!* ``` + +## Summary + +1. To extend a class: `class Child extends Parent`: + - That means `Child.prototype.__proto__` will be `Parent.prototype`, so methods are inherited. +2. When overriding a constructor: + - We must call parent constructor as `super()` in `Child` constructor before using `this`. +3. When overriding another method: + - We can use `super.method()` in a `Child` method to call `Parent` method. +4. Internals: + - Methods remember their class/object in the internal `[[HomeObject]]` property. That's how `super` resolves parent methods. + - So it's not safe to copy a method with `super` from one object to another. + +Also: +- Arrow functions don't have their own `this` or `super`, so they transparently fit into the surrounding context. diff --git a/1-js/09-classes/02-class-inheritance/super-homeobject-wrong.svg b/1-js/09-classes/02-class-inheritance/super-homeobject-wrong.svg new file mode 100644 index 000000000..f6450ddc4 --- /dev/null +++ b/1-js/09-classes/02-class-inheritance/super-homeobject-wrong.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="395" height="192" viewBox="0 0 395 192"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="inheritance" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="super-homeobject-wrong.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M335 30v28h-98V30h98z"/><text id="sayHi" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="247" y="48">sayHi</tspan></text><text id="plant" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="241" y="22">plant</tspan></text><path id="Rectangle-1-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M335 143v28h-98v-28h98z"/><text id="sayHi" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="247" y="162">sayHi</tspan></text><text id="tree" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="241" y="135">tree</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M287.5 70.5l7 14h-6v28h-2v-28h-6l7-14z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M121.025 150.55l-.022 6.001 124 .449 1 .004-.007 2-1-.004-124-.449-.021 6L107 157.5l14.025-6.95z"/><path id="Rectangle-1-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M8 30h98v28H8z"/><text id="sayHi" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="18" y="48">sayHi</tspan></text><path id="Rectangle-1-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M8 143h98v28H8z"/><text id="animal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="22">animal</tspan></text><text id="rabbit" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="131">rabbit</tspan></text><path id="Line-Copy-4" fill="#C06334" fill-rule="nonzero" d="M58.5 72.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[HomeObject]]" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="115" y="142">[[HomeObject]]</tspan></text><text id="sayHi" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="16" y="162">sayHi</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/09-classes/02-class-inheritance/3-class-extend-object/rabbit-extends-object.svg b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/rabbit-extends-object.svg similarity index 54% rename from 1-js/09-classes/02-class-inheritance/3-class-extend-object/rabbit-extends-object.svg rename to 1-js/09-classes/03-static-properties-methods/3-class-extend-object/rabbit-extends-object.svg index 8ab568291..8f948e447 100644 --- a/1-js/09-classes/02-class-inheritance/3-class-extend-object/rabbit-extends-object.svg +++ b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/rabbit-extends-object.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:1-js/09-classes/02-class-inheritance/3-class-extend-object/rabbit-extends-object.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="458px" height="344px" viewBox="0 0 458 344" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -64,4 +65,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="458" height="344" viewBox="0 0 458 344"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="inheritance" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="rabbit-extends-object.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M406 61v68H228V61h178z"/><text id="call:-function-bind:" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="238" y="80">call: function</tspan> <tspan x="238" y="95">bind: function</tspan> <tspan x="238" y="110">...</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M228 303h178v28H228z"/><text id="Function.prototype" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="227" y="53">Function.prototype</tspan></text><path id="Rectangle-1-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M406 202v28H228v-28h178z"/><text id="constructor" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="238" y="220">constructor</tspan></text><text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="227" y="194">Object</tspan></text><text id="Rabbit" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="233" y="292">Rabbit</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M316.5 143.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[Prototype]]" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="167">[[Prototype]]</tspan></text><path id="Line-Copy-3" fill="#C06334" fill-rule="nonzero" d="M316.5 247.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[Prototype]]-Copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="271">[[Prototype]]</tspan></text><text id="constructor-copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="236" y="321">constructor</tspan></text><path id="Rectangle-1-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M8 61h178v68H8z"/><text id="call:-function-bind:-copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="18" y="80">call: function</tspan> <tspan x="18" y="95">bind: function</tspan> <tspan x="18" y="110">...</tspan></text><path id="Rectangle-1-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M8 201h178v28H8z"/><text id="Function.prototype-Copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7" y="53">Function.prototype</tspan></text><text id="Rabbit-Copy" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="13" y="190">Rabbit</tspan></text><path id="Line-Copy-4" fill="#C06334" fill-rule="nonzero" d="M96.5 145.5l7 14h-6v28h-2v-28h-6l7-14z"/><text id="[[Prototype]]-Copy-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="106" y="169">[[Prototype]]</tspan></text><text id="constructor-copy-2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="16" y="219">constructor</tspan></text><text id="class-Rabbit" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="14" y="18">class Rabbit</tspan></text><text id="class-Rabbit-extends" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="224" y="18">class Rabbit extends Object</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:1-js/09-classes/03-static-properties-methods/3-class-extend-object/rabbit-extends-object.svg diff --git a/1-js/09-classes/02-class-inheritance/3-class-extend-object/solution.md b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/solution.md similarity index 76% rename from 1-js/09-classes/02-class-inheritance/3-class-extend-object/solution.md rename to 1-js/09-classes/03-static-properties-methods/3-class-extend-object/solution.md index fa26ec834..cb9829ce0 100644 --- a/1-js/09-classes/02-class-inheritance/3-class-extend-object/solution.md +++ b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/solution.md @@ -21,14 +21,14 @@ alert( rabbit.hasOwnProperty('name') ); // true But that's not all yet. -Even after the fix, there's still important difference in `"class Rabbit extends Object"` versus `class Rabbit`. +Even after the fix, there's still an important difference between `"class Rabbit extends Object"` and `class Rabbit`. As we know, the "extends" syntax sets up two prototypes: 1. Between `"prototype"` of the constructor functions (for methods). -2. Between the constructor functions itself (for static methods). +2. Between the constructor functions themselves (for static methods). -In our case, for `class Rabbit extends Object` it means: +In the case of `class Rabbit extends Object` it means: ```js run class Rabbit extends Object {} @@ -37,7 +37,7 @@ alert( Rabbit.prototype.__proto__ === Object.prototype ); // (1) true alert( Rabbit.__proto__ === Object ); // (2) true ``` -So `Rabbit` now provides access to static methods of `Object` via `Rabbit`, like this: +So `Rabbit` now provides access to the static methods of `Object` via `Rabbit`, like this: ```js run class Rabbit extends Object {} @@ -67,7 +67,7 @@ alert ( Rabbit.getOwnPropertyNames({a: 1, b: 2})); // Error So `Rabbit` doesn't provide access to static methods of `Object` in that case. -By the way, `Function.prototype` has "generic" function methods, like `call`, `bind` etc. They are ultimately available in both cases, because for the built-in `Object` constructor, `Object.__proto__ === Function.prototype`. +By the way, `Function.prototype` also has "generic" function methods, like `call`, `bind` etc. They are ultimately available in both cases, because for the built-in `Object` constructor, `Object.__proto__ === Function.prototype`. Here's the picture: diff --git a/1-js/09-classes/02-class-inheritance/3-class-extend-object/task.md b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/task.md similarity index 88% rename from 1-js/09-classes/02-class-inheritance/3-class-extend-object/task.md rename to 1-js/09-classes/03-static-properties-methods/3-class-extend-object/task.md index ca6628edf..1d0f98a74 100644 --- a/1-js/09-classes/02-class-inheritance/3-class-extend-object/task.md +++ b/1-js/09-classes/03-static-properties-methods/3-class-extend-object/task.md @@ -1,4 +1,4 @@ -importance: 5 +importance: 3 --- @@ -19,7 +19,6 @@ let rabbit = new Rabbit("Rab"); *!* // hasOwnProperty method is from Object.prototype -// rabbit.__proto__ === Object.prototype alert( rabbit.hasOwnProperty('name') ); // true */!* ``` @@ -39,5 +38,5 @@ class Rabbit extends Object { let rabbit = new Rabbit("Rab"); -alert( rabbit.hasOwnProperty('name') ); // true +alert( rabbit.hasOwnProperty('name') ); // Error ``` diff --git a/1-js/09-classes/03-static-properties-methods/article.md b/1-js/09-classes/03-static-properties-methods/article.md index 6ccf4fd71..0cfc07797 100644 --- a/1-js/09-classes/03-static-properties-methods/article.md +++ b/1-js/09-classes/03-static-properties-methods/article.md @@ -1,9 +1,9 @@ # Static properties and methods -We can also assign a method to the class function, not to its `"prototype"`. Such methods are called *static*. +We can also assign a method to the class as a whole. Such methods are called *static*. -An example: +In a class declaration, they are prepended by `static` keyword, like this: ```js run class User { @@ -17,21 +17,25 @@ class User { User.staticMethod(); // true ``` -That actually does the same as assigning it as a function property: +That actually does the same as assigning it as a property directly: -```js -function User() { } +```js run +class User { } User.staticMethod = function() { alert(this === User); }; + +User.staticMethod(); // true ``` -The value of `this` inside `User.staticMethod()` is the class constructor `User` itself (the "object before dot" rule). +The value of `this` in `User.staticMethod()` call is the class constructor `User` itself (the "object before dot" rule). + +Usually, static methods are used to implement functions that belong to the class as a whole, but not to any particular object of it. -Usually, static methods are used to implement functions that belong to the class, but not to any particular object of it. +For instance, we have `Article` objects and need a function to compare them. -For instance, we have `Article` objects and need a function to compare them. The natural choice would be `Article.compare`, like this: +A natural solution would be to add `Article.compare` static method: ```js run class Article { @@ -49,8 +53,8 @@ class Article { // usage let articles = [ - new Article("Mind", new Date(2019, 1, 1)), - new Article("Body", new Date(2019, 0, 1)), + new Article("HTML", new Date(2019, 1, 1)), + new Article("CSS", new Date(2019, 0, 1)), new Article("JavaScript", new Date(2019, 11, 1)) ]; @@ -58,20 +62,22 @@ let articles = [ articles.sort(Article.compare); */!* -alert( articles[0].title ); // Body +alert( articles[0].title ); // CSS ``` -Here `Article.compare` stands "over" the articles, as a means to compare them. It's not a method of an article, but rather of the whole class. +Here `Article.compare` method stands "above" articles, as a means to compare them. It's not a method of an article, but rather of the whole class. -Another example would be a so-called "factory" method. Imagine, we need few ways to create an article: +Another example would be a so-called "factory" method. + +Let's say, we need multiple ways to create an article: 1. Create by given parameters (`title`, `date` etc). 2. Create an empty article with today's date. -3. ... +3. ...or else somehow. The first way can be implemented by the constructor. And for the second one we can make a static method of the class. -Like `Article.createTodays()` here: +Such as `Article.createTodays()` here: ```js run class Article { @@ -90,7 +96,7 @@ class Article { let article = Article.createTodays(); -alert( article.title ); // Todays digest +alert( article.title ); // Today's digest ``` Now every time we need to create a today's digest, we can call `Article.createTodays()`. Once again, that's not a method of an article, but a method of the whole class. @@ -99,15 +105,26 @@ Static methods are also used in database-related classes to search/save/remove e ```js // assuming Article is a special class for managing articles -// static method to remove the article: +// static method to remove the article by id: Article.remove({id: 12345}); ``` +````warn header="Static methods aren't available for individual objects" +Static methods are callable on classes, not on individual objects. + +E.g. such code won't work: + +```js +// ... +article.createTodays(); /// Error: article.createTodays is not a function +``` +```` + ## Static properties [recent browser=Chrome] -Static properties are also possible, just like regular class properties: +Static properties are also possible, they look like regular class properties, but prepended by `static`: ```js run class Article { @@ -123,14 +140,15 @@ That is the same as a direct assignment to `Article`: Article.publisher = "Ilya Kantor"; ``` -## Statics and inheritance +## Inheritance of static properties and methods [#statics-and-inheritance] -Statics are inherited, we can access `Parent.method` as `Child.method`. +Static properties and methods are inherited. -For instance, `Animal.compare` in the code below is inherited and accessible as `Rabbit.compare`: +For instance, `Animal.compare` and `Animal.planet` in the code below are inherited and accessible as `Rabbit.compare` and `Rabbit.planet`: ```js run class Animal { + static planet = "Earth"; constructor(name, speed) { this.speed = speed; @@ -167,38 +185,47 @@ rabbits.sort(Rabbit.compare); */!* rabbits[0].run(); // Black Rabbit runs with speed 5. + +alert(Rabbit.planet); // Earth ``` -Now we can call `Rabbit.compare` assuming that the inherited `Animal.compare` will be called. +Now when we call `Rabbit.compare`, the inherited `Animal.compare` will be called. -How does it work? Again, using prototypes. As you might have already guessed, extends also gives `Rabbit` the `[[Prototype]]` reference to `Animal`. +How does it work? Again, using prototypes. As you might have already guessed, `extends` gives `Rabbit` the `[[Prototype]]` reference to `Animal`. + +<<<<<<< HEAD  +======= +So, `Rabbit extends Animal` creates two `[[Prototype]]` references: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b + +1. `Rabbit` function prototypally inherits from `Animal` function. +2. `Rabbit.prototype` prototypally inherits from `Animal.prototype`. -So, `Rabbit` function now inherits from `Animal` function. And `Animal` function normally has `[[Prototype]]` referencing `Function.prototype`, because it doesn't `extend` anything. +As a result, inheritance works both for regular and static methods. -Here, let's check that: +Here, let's check that by code: ```js run class Animal {} class Rabbit extends Animal {} -// for static properties and methods +// for statics alert(Rabbit.__proto__ === Animal); // true -// and the next step is Function.prototype -alert(Animal.__proto__ === Function.prototype); // true - -// that's in addition to the "normal" prototype chain for object methods -alert(Rabbit.prototype.__proto__ === Animal.prototype); +// for regular methods +alert(Rabbit.prototype.__proto__ === Animal.prototype); // true ``` -This way `Rabbit` has access to all static methods of `Animal`. - ## Summary -Static methods are used for the functionality that doesn't relate to a concrete class instance, doesn't require an instance to exist, but rather belongs to the class as a whole, like `Article.compare` -- a generic method to compare two articles. +Static methods are used for the functionality that belongs to the class "as a whole". It doesn't relate to a concrete class instance. + +For example, a method for comparison `Article.compare(article1, article2)` or a factory method `Article.createTodays()`. + +They are labeled by the word `static` in class declaration. Static properties are used when we'd like to store class-level data, also not bound to an instance. @@ -214,13 +241,13 @@ class MyClass { } ``` -That's technically the same as assigning to the class itself: +Technically, static declaration is the same as assigning to the class itself: ```js MyClass.property = ... MyClass.method = ... ``` -Static properties are inherited. +Static properties and methods are inherited. -Technically, for `class B extends A` the prototype of the class `B` itself points to `A`: `B.[[Prototype]] = A`. So if a field is not found in `B`, the search continues in `A`. +For `class B extends A` the prototype of the class `B` itself points to `A`: `B.[[Prototype]] = A`. So if a field is not found in `B`, the search continues in `A`. diff --git a/1-js/09-classes/04-private-protected-properties-methods/article.md b/1-js/09-classes/04-private-protected-properties-methods/article.md index ea6753419..91efb89ee 100644 --- a/1-js/09-classes/04-private-protected-properties-methods/article.md +++ b/1-js/09-classes/04-private-protected-properties-methods/article.md @@ -48,16 +48,16 @@ So, all we need to use an object is to know its external interface. We may be co That was a general introduction. -In JavaScript, there are three types of properties and members: +In JavaScript, there are two types of object fields (properties and methods): -- Public: accessible from anywhere. They comprise the external interface. Till now we were only using public properties and methods. +- Public: accessible from anywhere. They comprise the external interface. Until now we were only using public properties and methods. - Private: accessible only from inside the class. These are for the internal interface. -In many other languages there also exist "protected" fields: accessible only from inside the class and those extending it. They are also useful for the internal interface. They are in a sense more widespread than private ones, because we usually want inheriting classes to gain access to properly do the extension. +In many other languages there also exist "protected" fields: accessible only from inside the class and those extending it (like private, but plus access from inheriting classes). They are also useful for the internal interface. They are in a sense more widespread than private ones, because we usually want inheriting classes to gain access to them. Protected fields are not implemented in JavaScript on the language level, but in practice they are very convenient, so they are emulated. -In the next step we'll make a coffee machine in JavaScript with all these types of properties. A coffee machine has a lot of details, we won't model them to stay simple (though we could). +Now we'll make a coffee machine in JavaScript with all these types of properties. A coffee machine has a lot of details, we won't model them to stay simple (though we could). ## Protecting "waterAmount" @@ -87,7 +87,7 @@ Let's change `waterAmount` property to protected to have more control over it. F **Protected properties are usually prefixed with an underscore `_`.** -That is not enforced on the language level, but there's a convention that such properties and methods should not be accessed from the outside. Most programmers follow it. +That is not enforced on the language level, but there's a well-known convention between programmers that such properties and methods should not be accessed from the outside. So our property will be called `_waterAmount`: @@ -96,7 +96,9 @@ class CoffeeMachine { _waterAmount = 0; set waterAmount(value) { - if (value < 0) throw new Error("Negative water"); + if (value < 0) { + value = 0; + } this._waterAmount = value; } @@ -114,10 +116,10 @@ class CoffeeMachine { let coffeeMachine = new CoffeeMachine(100); // add water -coffeeMachine.waterAmount = -10; // Error: Negative water +coffeeMachine.waterAmount = -10; // _waterAmount will become 0, not -10 ``` -Now the access is under control, so setting the water below zero fails. +Now the access is under control, so setting the water amount below zero becomes impossible. ## Read-only "power" @@ -159,21 +161,21 @@ class CoffeeMachine { _waterAmount = 0; *!*setWaterAmount(value)*/!* { - if (value < 0) throw new Error("Negative water"); + if (value < 0) value = 0; this._waterAmount = value; } *!*getWaterAmount()*/!* { - return this.waterAmount; + return this._waterAmount; } } new CoffeeMachine().setWaterAmount(100); ``` -That looks a bit longer, but functions are more flexible. They can accept multiple arguments (even if we don't need them right now). So, for the future, just in case we need to refactor something, functions are a safer choice. +That looks a bit longer, but functions are more flexible. They can accept multiple arguments (even if we don't need them right now). -Surely, there's a tradeoff. On the other hand, get/set syntax is shorter, so ultimately there's no strict rule, it's up to you to decide. +On the other hand, get/set syntax is shorter, so ultimately there's no strict rule, it's up to you to decide. ```` ```smart header="Protected fields are inherited" @@ -190,32 +192,23 @@ There's a finished JavaScript proposal, almost in the standard, that provides la Privates should start with `#`. They are only accessible from inside the class. -For instance, here we add a private `#waterLimit` property and extract the water-checking logic into a separate method: +For instance, here's a private `#waterLimit` property and the water-checking private method `#fixWaterAmount`: -```js +```js run class CoffeeMachine { *!* #waterLimit = 200; */!* *!* - #checkWater(value) { - if (value < 0) throw new Error("Negative water"); - if (value > this.#waterLimit) throw new Error("Too much water"); + #fixWaterAmount(value) { + if (value < 0) return 0; + if (value > this.#waterLimit) return this.#waterLimit; } */!* - _waterAmount = 0; - - set waterAmount(value) { -*!* - this.#checkWater(value); -*/!* - this._waterAmount = value; - } - - get waterAmount() { - return this.waterAmount; + setWaterAmount(value) { + this.#waterLimit = this.#fixWaterAmount(value); } } @@ -223,11 +216,10 @@ class CoffeeMachine { let coffeeMachine = new CoffeeMachine(); *!* -coffeeMachine.#checkWater(); // Error +// can't access privates from outside of the class +coffeeMachine.#fixWaterAmount(123); // Error coffeeMachine.#waterLimit = 1000; // Error */!* - -coffeeMachine.waterAmount = 100; // Works ``` On the language level, `#` is a special sign that the field is private. We can't access it from outside or from inheriting classes. @@ -246,7 +238,7 @@ class CoffeeMachine { } set waterAmount(value) { - if (value < 0) throw new Error("Negative water"); + if (value < 0) value = 0; this.#waterAmount = value; } } @@ -262,7 +254,7 @@ Unlike protected ones, private fields are enforced by the language itself. That' But if we inherit from `CoffeeMachine`, then we'll have no direct access to `#waterAmount`. We'll need to rely on `waterAmount` getter/setter: ```js -class CoffeeMachine extends CoffeeMachine() { +class MegaCoffeeMachine extends CoffeeMachine { method() { *!* alert( this.#waterAmount ); // Error: can only access from CoffeeMachine @@ -271,19 +263,19 @@ class CoffeeMachine extends CoffeeMachine() { } ``` -In many scenarios such limitation is too severe. If we extend a `CoffeeMachine`, we may have legitimate reason to access its internals. That's why protected fields are used most of the time, even though they are not supported by the language syntax. +In many scenarios such limitation is too severe. If we extend a `CoffeeMachine`, we may have legitimate reasons to access its internals. That's why protected fields are used more often, even though they are not supported by the language syntax. -````warn +````warn header="Private fields are not available as this[name]" Private fields are special. -Remember, usually we can access fields by this[name]: +As we know, usually we can access fields using `this[name]`: ```js class User { ... sayHi() { let fieldName = "name"; - alert(`Hello, ${this[fieldName]}`); + alert(`Hello, ${*!*this[fieldName]*/!*}`); } } ``` @@ -293,11 +285,11 @@ With private fields that's impossible: `this['#name']` doesn't work. That's a sy ## Summary -In terms of OOP, delimiting of the internal interface from the external one is called [encapsulation]("https://en.wikipedia.org/wiki/Encapsulation_(computer_programming)"). +In terms of OOP, delimiting of the internal interface from the external one is called [encapsulation](https://en.wikipedia.org/wiki/Encapsulation_(computer_programming)). It gives the following benefits: -Protection for users, so that they don't shoot themselves in the feet +Protection for users, so that they don't shoot themselves in the foot : Imagine, there's a team of developers using a coffee machine. It was made by the "Best CoffeeMachine" company, and works fine, but a protective cover was removed. So the internal interface is exposed. All developers are civilized -- they use the coffee machine as intended. But one of them, John, decided that he's the smartest one, and made some tweaks in the coffee machine internals. So the coffee machine failed two days later. @@ -309,22 +301,22 @@ Protection for users, so that they don't shoot themselves in the feet Supportable : The situation in programming is more complex than with a real-life coffee machine, because we don't just buy it once. The code constantly undergoes development and improvement. - **If we strictly delimit the internal interface, then the developer of the class can freely change its internal properties and methods, even without informing the users..** + **If we strictly delimit the internal interface, then the developer of the class can freely change its internal properties and methods, even without informing the users.** - It's much easier to develop, if you know that certain methods can be renamed, their parameters can be changed, and even removed, because no external code depends on them. + If you're a developer of such class, it's great to know that private methods can be safely renamed, their parameters can be changed, and even removed, because no external code depends on them. - For users, when a new version comes out, it may be a total overhaul, but still simple to upgrade if the external interface is the same. + For users, when a new version comes out, it may be a total overhaul internally, but still simple to upgrade if the external interface is the same. Hiding complexity -: People adore to use things that are simple. At least from outside. What's inside is a different thing. +: People adore using things that are simple. At least from outside. What's inside is a different thing. Programmers are not an exception. **It's always convenient when implementation details are hidden, and a simple, well-documented external interface is available.** -To hide internal interface we use either protected or public properties: +To hide an internal interface we use either protected or private properties: - Protected fields start with `_`. That's a well-known convention, not enforced at the language level. Programmers should only access a field starting with `_` from its class and classes inheriting from it. -- Private fields start with `#`. JavaScript makes sure we only can access those from inside the class. +- Private fields start with `#`. JavaScript makes sure we can only access those from inside the class. Right now, private fields are not well-supported among browsers, but can be polyfilled. diff --git a/1-js/09-classes/05-extend-natives/article.md b/1-js/09-classes/05-extend-natives/article.md index 4cf1c2bdd..28b4c6eb6 100644 --- a/1-js/09-classes/05-extend-natives/article.md +++ b/1-js/09-classes/05-extend-natives/article.md @@ -21,21 +21,20 @@ alert(filteredArr); // 10, 50 alert(filteredArr.isEmpty()); // false ``` -Please note a very interesting thing. Built-in methods like `filter`, `map` and others -- return new objects of exactly the inherited type. They rely on the `constructor` property to do so. +Please note a very interesting thing. Built-in methods like `filter`, `map` and others -- return new objects of exactly the inherited type `PowerArray`. Their internal implementation uses the object's `constructor` property for that. In the example above, ```js arr.constructor === PowerArray ``` -So when `arr.filter()` is called, it internally creates the new array of results exactly as `new PowerArray`. -That's actually very cool, because we can keep using `PowerArray` methods further on the result. +When `arr.filter()` is called, it internally creates the new array of results using exactly `arr.constructor`, not basic `Array`. That's actually very cool, because we can keep using `PowerArray` methods further on the result. Even more, we can customize that behavior. -There's a special static getter `Symbol.species`, if exists, it returns the constructor to use in such cases. +We can add a special static getter `Symbol.species` to the class. If it exists, it should return the constructor that JavaScript will use internally to create new entities in `map`, `filter` and so on. -If we'd like built-in methods like `map`, `filter` will return regular arrays, we can return `Array` in `Symbol.species`, like here: +If we'd like built-in methods like `map` or `filter` to return regular arrays, we can return `Array` in `Symbol.species`, like here: ```js run class PowerArray extends Array { @@ -65,18 +64,26 @@ alert(filteredArr.isEmpty()); // Error: filteredArr.isEmpty is not a function As you can see, now `.filter` returns `Array`. So the extended functionality is not passed any further. +```smart header="Other collections work similarly" +Other collections, such as `Map` and `Set`, work alike. They also use `Symbol.species`. +``` + ## No static inheritance in built-ins Built-in objects have their own static methods, for instance `Object.keys`, `Array.isArray` etc. -And we've already been talking about native classes extending each other: `Array.[[Prototype]] = Object`. +As we already know, native classes extend each other. For instance, `Array` extends `Object`. -But statics are an exception. Built-in classes don't inherit static properties from each other. +Normally, when one class extends another, both static and non-static methods are inherited. That was thoroughly explained in the article [](info:static-properties-methods#statics-and-inheritance). -In other words, the prototype of built-in constructor `Array` does not point to `Object`. This way `Array` and `Date` do not have `Array.keys` or `Date.keys`. And that feels natural. +But built-in classes are an exception. They don't inherit statics from each other. + +For example, both `Array` and `Date` inherit from `Object`, so their instances have methods from `Object.prototype`. But `Array.[[Prototype]]` does not reference `Object`, so there's no, for instance, `Array.keys()` (or `Date.keys()`) static method. Here's the picture structure for `Date` and `Object`:  -Note, there's no link between `Date` and `Object`. Both `Object` and `Date` exist independently. `Date.prototype` inherits from `Object.prototype`, but that's all. +As you can see, there's no link between `Date` and `Object`. They are independent, only `Date.prototype` inherits from `Object.prototype`. + +That's an important difference of inheritance between built-in objects compared to what we get with `extends`. diff --git a/1-js/09-classes/06-instanceof/1-strange-instanceof/task.md b/1-js/09-classes/06-instanceof/1-strange-instanceof/task.md index e9481912a..5b8dc7de3 100644 --- a/1-js/09-classes/06-instanceof/1-strange-instanceof/task.md +++ b/1-js/09-classes/06-instanceof/1-strange-instanceof/task.md @@ -4,7 +4,7 @@ importance: 5 # Strange instanceof -Why `instanceof` below returns `true`? We can easily see that `a` is not created by `B()`. +In the code below, why does `instanceof` return `true`? We can easily see that `a` is not created by `B()`. ```js run function A() {} diff --git a/1-js/09-classes/06-instanceof/article.md b/1-js/09-classes/06-instanceof/article.md index 50adef552..f9db989ca 100644 --- a/1-js/09-classes/06-instanceof/article.md +++ b/1-js/09-classes/06-instanceof/article.md @@ -2,7 +2,7 @@ The `instanceof` operator allows to check whether an object belongs to a certain class. It also takes inheritance into account. -Such a check may be necessary in many cases, here we'll use it for building a *polymorphic* function, the one that treats arguments differently depending on their type. +Such a check may be necessary in many cases. For example, it can be used for building a *polymorphic* function, the one that treats arguments differently depending on their type. ## The instanceof operator [#ref-instanceof] @@ -11,7 +11,7 @@ The syntax is: obj instanceof Class ``` -It returns `true` if `obj` belongs to the `Class` (or a class inheriting from it). +It returns `true` if `obj` belongs to the `Class` or a class inheriting from it. For instance: @@ -44,16 +44,19 @@ alert( arr instanceof Array ); // true alert( arr instanceof Object ); // true ``` -Please note that `arr` also belongs to the `Object` class. That's because `Array` prototypally inherits from `Object`. +Please note that `arr` also belongs to the `Object` class. That's because `Array` prototypically inherits from `Object`. -The `instanceof` operator examines the prototype chain for the check, and is also fine-tunable using the static method `Symbol.hasInstance`. +Normally, `instanceof` examines the prototype chain for the check. We can also set a custom logic in the static method `Symbol.hasInstance`. The algorithm of `obj instanceof Class` works roughly as follows: -1. If there's a static method `Symbol.hasInstance`, then use it. Like this: +1. If there's a static method `Symbol.hasInstance`, then just call it: `Class[Symbol.hasInstance](obj)`. It should return either `true` or `false`, and we're done. That's how we can customize the behavior of `instanceof`. + + For example: ```js run - // assume anything that canEat is an animal + // setup instanceOf check that assumes that + // anything with canEat property is an animal class Animal { static [Symbol.hasInstance](obj) { if (obj.canEat) return true; @@ -61,22 +64,25 @@ The algorithm of `obj instanceof Class` works roughly as follows: } let obj = { canEat: true }; + alert(obj instanceof Animal); // true: Animal[Symbol.hasInstance](obj) is called ``` -2. Most classes do not have `Symbol.hasInstance`. In that case, check if `Class.prototype` equals to one of prototypes in the `obj` prototype chain. +2. Most classes do not have `Symbol.hasInstance`. In that case, the standard logic is used: `obj instanceOf Class` checks whether `Class.prototype` is equal to one of the prototypes in the `obj` prototype chain. - In other words, compare: + In other words, compare one after another: ```js - obj.__proto__ === Class.prototype - obj.__proto__.__proto__ === Class.prototype - obj.__proto__.__proto__.__proto__ === Class.prototype + obj.__proto__ === Class.prototype? + obj.__proto__.__proto__ === Class.prototype? + obj.__proto__.__proto__.__proto__ === Class.prototype? ... + // if any answer is true, return true + // otherwise, if we reached the end of the chain, return false ``` - In the example above `Rabbit.prototype === rabbit.__proto__`, so that gives the answer immediately. + In the example above `rabbit.__proto__ === Rabbit.prototype`, so that gives the answer immediately. - In the case of an inheritance, `rabbit` is an instance of the parent class as well: + In the case of an inheritance, the match will be at the second step: ```js run class Animal {} @@ -86,8 +92,11 @@ The algorithm of `obj instanceof Class` works roughly as follows: *!* alert(rabbit instanceof Animal); // true */!* - // rabbit.__proto__ === Rabbit.prototype + + // rabbit.__proto__ === Animal.prototype (no match) + *!* // rabbit.__proto__.__proto__ === Animal.prototype (match!) + */!* ``` Here's the illustration of what `rabbit instanceof Animal` compares with `Animal.prototype`: @@ -96,9 +105,9 @@ Here's the illustration of what `rabbit instanceof Animal` compares with `Animal By the way, there's also a method [objA.isPrototypeOf(objB)](mdn:js/object/isPrototypeOf), that returns `true` if `objA` is somewhere in the chain of prototypes for `objB`. So the test of `obj instanceof Class` can be rephrased as `Class.prototype.isPrototypeOf(obj)`. -That's funny, but the `Class` constructor itself does not participate in the check! Only the chain of prototypes and `Class.prototype` matters. +It's funny, but the `Class` constructor itself does not participate in the check! Only the chain of prototypes and `Class.prototype` matters. -That can lead to interesting consequences when `prototype` is changed. +That can lead to interesting consequences when a `prototype` property is changed after the object is created. Like here: @@ -115,9 +124,7 @@ alert( rabbit instanceof Rabbit ); // false */!* ``` -That's one of the reasons to avoid changing `prototype`. Just to keep safe. - -## Bonus: Object toString for the type +## Bonus: Object.prototype.toString for the type We already know that plain objects are converted to string as `[object Object]`: @@ -150,7 +157,7 @@ let objectToString = Object.prototype.toString; // what type is this? let arr = []; -alert( objectToString.call(arr) ); // [object Array] +alert( objectToString.call(arr) ); // [object *!*Array*/!*] ``` Here we used [call](mdn:js/function/call) as described in the chapter [](info:call-apply-decorators) to execute the function `objectToString` in the context `this=arr`. @@ -179,11 +186,11 @@ let user = { alert( {}.toString.call(user) ); // [object User] ``` -For most environment-specific objects, there is such a property. Here are few browser specific examples: +For most environment-specific objects, there is such a property. Here are some browser specific examples: ```js run // toStringTag for the environment-specific object and class: -alert( window[Symbol.toStringTag]); // window +alert( window[Symbol.toStringTag]); // Window alert( XMLHttpRequest.prototype[Symbol.toStringTag] ); // XMLHttpRequest alert( {}.toString.call(window) ); // [object Window] @@ -194,11 +201,11 @@ As you can see, the result is exactly `Symbol.toStringTag` (if exists), wrapped At the end we have "typeof on steroids" that not only works for primitive data types, but also for built-in objects and even can be customized. -It can be used instead of `instanceof` for built-in objects when we want to get the type as a string rather than just to check. +We can use `{}.toString.call` instead of `instanceof` for built-in objects when we want to get the type as a string rather than just to check. ## Summary -Let's recap the type-checking methods that we know: +Let's summarize the type-checking methods that we know: | | works for | returns | |---------------|-------------|---------------| diff --git a/1-js/09-classes/07-mixins/article.md b/1-js/09-classes/07-mixins/article.md index 7b6d9ebad..21c1097d0 100644 --- a/1-js/09-classes/07-mixins/article.md +++ b/1-js/09-classes/07-mixins/article.md @@ -2,19 +2,19 @@ In JavaScript we can only inherit from a single object. There can be only one `[[Prototype]]` for an object. And a class may extend only one other class. -But sometimes that feels limiting. For instance, I have a class `StreetSweeper` and a class `Bicycle`, and want to make a `StreetSweepingBicycle`. +But sometimes that feels limiting. For instance, we have a class `StreetSweeper` and a class `Bicycle`, and want to make their mix: a `StreetSweepingBicycle`. -Or, talking about programming, we have a class `Renderer` that implements templating and a class `EventEmitter` that implements event handling, and want to merge these functionalities together with a class `Page`, to make a page that can use templates and emit events. +Or we have a class `User` and a class `EventEmitter` that implements event generation, and we'd like to add the functionality of `EventEmitter` to `User`, so that our users can emit events. There's a concept that can help here, called "mixins". -As defined in Wikipedia, a [mixin](https://en.wikipedia.org/wiki/Mixin) is a class that contains methods for use by other classes without having to be the parent class of those other classes. +As defined in Wikipedia, a [mixin](https://en.wikipedia.org/wiki/Mixin) is a class containing methods that can be used by other classes without a need to inherit from it. In other words, a *mixin* provides methods that implement a certain behavior, but we do not use it alone, we use it to add the behavior to other classes. ## A mixin example -The simplest way to make a mixin in JavaScript is to make an object with useful methods, so that we can easily merge them into a prototype of any class. +The simplest way to implement a mixin in JavaScript is to make an object with useful methods, so that we can easily merge them into a prototype of any class. For instance here the mixin `sayHiMixin` is used to add some "speech" for `User`: @@ -47,7 +47,7 @@ Object.assign(User.prototype, sayHiMixin); new User("Dude").sayHi(); // Hello Dude! ``` -There's no inheritance, but a simple method copying. So `User` may extend some other class and also include the mixin to "mix-in" the additional methods, like this: +There's no inheritance, but a simple method copying. So `User` may inherit from another class and also include the mixin to "mix-in" the additional methods, like this: ```js class User extends Person { @@ -69,16 +69,16 @@ let sayMixin = { }; let sayHiMixin = { - __proto__: sayMixin, // (or we could use Object.create to set the prototype here) + __proto__: sayMixin, // (or we could use Object.setPrototypeOf to set the prototype here) sayHi() { *!* // call parent method */!* - super.say(`Hello ${this.name}`); + super.say(`Hello ${this.name}`); // (*) }, sayBye() { - super.say(`Bye ${this.name}`); + super.say(`Bye ${this.name}`); // (*) } }; @@ -95,27 +95,35 @@ Object.assign(User.prototype, sayHiMixin); new User("Dude").sayHi(); // Hello Dude! ``` -Please note that the call to the parent method `super.say()` from `sayHiMixin` looks for the method in the prototype of that mixin, not the class. +Please note that the call to the parent method `super.say()` from `sayHiMixin` (at lines labelled with `(*)`) looks for the method in the prototype of that mixin, not the class. +<<<<<<< HEAD  +======= +Here's the diagram (see the right part): +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -That's because methods from `sayHiMixin` have `[[HomeObject]]` set to it. So `super` actually means `sayHiMixin.__proto__`, not `User.__proto__`. + + +That's because methods `sayHi` and `sayBye` were initially created in `sayHiMixin`. So even though they got copied, their `[[HomeObject]]` internal property references `sayHiMixin`, as shown in the picture above. + +As `super` looks for parent methods in `[[HomeObject]].[[Prototype]]`, that means it searches `sayHiMixin.[[Prototype]]`. ## EventMixin Now let's make a mixin for real life. -The important feature of many objects is working with events. +An important feature of many browser objects (for instance) is that they can generate events. Events are a great way to "broadcast information" to anyone who wants it. So let's make a mixin that allows us to easily add event-related functions to any class/object. -That is: an object should have a method to "generate an event" when something important happens to it, and other objects should be able to "listen" to such events. +- The mixin will provide a method `.trigger(name, [...data])` to "generate an event" when something important happens to it. The `name` argument is a name of the event, optionally followed by additional arguments with event data. +- Also the method `.on(name, handler)` that adds `handler` function as the listener to events with the given name. It will be called when an event with the given `name` triggers, and get the arguments from the `.trigger` call. +- ...And the method `.off(name, handler)` that removes the `handler` listener. -An event must have a name and, optionally, bundle some additional data. +After adding the mixin, an object `user` will be able to generate an event `"login"` when the visitor logs in. And another object, say, `calendar` may want to listen for such events to load the calendar for the logged-in person. -For instance, an object `user` can generate an event `"login"` when the visitor logs in. And another object `calendar` may want to receive such events to load the calendar for the logged-in person. +Or, a `menu` can generate the event `"select"` when a menu item is selected, and other objects may assign handlers to react on that event. And so on. -Or, a `menu` can generate the event `"select"` when a menu item is selected, and other objects may want to get that information and react on that event. - -Events is a way to "share information" with anyone who wants it. They can be useful in any class, so let's make a mixin for them: +Here's the code: ```js run let eventMixin = { @@ -136,7 +144,7 @@ let eventMixin = { * menu.off('select', handler) */ off(eventName, handler) { - let handlers = this._eventHandlers && this._eventHandlers[eventName]; + let handlers = this._eventHandlers?.[eventName]; if (!handlers) return; for (let i = 0; i < handlers.length; i++) { if (handlers[i] === handler) { @@ -146,11 +154,11 @@ let eventMixin = { }, /** - * Generate the event and attach the data to it + * Generate an event with the given name and data * this.trigger('select', data1, data2); */ trigger(eventName, ...args) { - if (!this._eventHandlers || !this._eventHandlers[eventName]) { + if (!this._eventHandlers?.[eventName]) { return; // no handlers for that event name } @@ -160,12 +168,10 @@ let eventMixin = { }; ``` -There are 3 methods here: - -1. `.on(eventName, handler)` -- assigns function `handler` to run when the event with that name happens. The handlers are stored in the `_eventHandlers` property. -2. `.off(eventName, handler)` -- removes the function from the handlers list. -3. `.trigger(eventName, ...args)` -- generates the event: all assigned handlers are called and `args` are passed as arguments to them. +- `.on(eventName, handler)` -- assigns function `handler` to run when the event with that name occurs. Technically, there's an `_eventHandlers` property that stores an array of handlers for each event name, and it just adds it to the list. +- `.off(eventName, handler)` -- removes the function from the handlers list. +- `.trigger(eventName, ...args)` -- generates the event: all handlers from `_eventHandlers[eventName]` are called, with a list of arguments `...args`. Usage: @@ -176,30 +182,31 @@ class Menu { this.trigger("select", value); } } -// Add the mixin +// Add the mixin with event-related methods Object.assign(Menu.prototype, eventMixin); let menu = new Menu(); -// call the handler on selection: +// add a handler, to be called on selection: *!* menu.on("select", value => alert(`Value selected: ${value}`)); */!* -// triggers the event => shows Value selected: 123 -menu.choose("123"); // value selected +// triggers the event => the handler above runs and shows: +// Value selected: 123 +menu.choose("123"); ``` -Now if we have the code interested to react on user selection, we can bind it with `menu.on(...)`. +Now, if we'd like any code to react to a menu selection, we can listen for it with `menu.on(...)`. -And the `eventMixin` can add such behavior to as many classes as we'd like, without interfering with the inheritance chain. +And `eventMixin` mixin makes it easy to add such behavior to as many classes as we'd like, without interfering with the inheritance chain. ## Summary *Mixin* -- is a generic object-oriented programming term: a class that contains methods for other classes. -Some other languages like e.g. python allow to create mixins using multiple inheritance. JavaScript does not support multiple inheritance, but mixins can be implemented by copying them into the prototype. +Some other languages allow multiple inheritance. JavaScript does not support multiple inheritance, but mixins can be implemented by copying methods into prototype. -We can use mixins as a way to augment a class by multiple behaviors, like event-handling as we have seen above. +We can use mixins as a way to augment a class by adding multiple behaviors, like event-handling as we have seen above. -Mixins may become a point of conflict if they occasionally overwrite native class methods. So generally one should think well about the naming for a mixin, to minimize such possibility. +Mixins may become a point of conflict if they accidentally overwrite existing class methods. So generally one should think well about the naming methods of a mixin, to minimize the probability of that happening. diff --git a/1-js/09-classes/07-mixins/head.html b/1-js/09-classes/07-mixins/head.html index 77ea38b20..20e3a6354 100644 --- a/1-js/09-classes/07-mixins/head.html +++ b/1-js/09-classes/07-mixins/head.html @@ -18,7 +18,7 @@ * menu.off('select', handler) */ off(eventName, handler) { - let handlers = this._eventHandlers && this._eventHandlers[eventName]; + let handlers = this._eventHandlers?.[eventName]; if (!handlers) return; for(let i = 0; i < handlers.length; i++) { if (handlers[i] == handler) { diff --git a/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/solution.md b/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/solution.md index 05ba72e00..ec0dabc9a 100644 --- a/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/solution.md +++ b/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/solution.md @@ -1,8 +1,8 @@ The difference becomes obvious when we look at the code inside a function. -The behavior is different if there's a "jump out" of `try..catch`. +The behavior is different if there's a "jump out" of `try...catch`. -For instance, when there's a `return` inside `try..catch`. The `finally` clause works in case of *any* exit from `try..catch`, even via the `return` statement: right after `try..catch` is done, but before the calling code gets the control. +For instance, when there's a `return` inside `try...catch`. The `finally` clause works in case of *any* exit from `try...catch`, even via the `return` statement: right after `try...catch` is done, but before the calling code gets the control. ```js run function f() { @@ -11,7 +11,7 @@ function f() { *!* return "result"; */!* - } catch (e) { + } catch (err) { /// ... } finally { alert('cleanup!'); @@ -28,11 +28,11 @@ function f() { try { alert('start'); throw new Error("an error"); - } catch (e) { + } catch (err) { // ... if("can't handle the error") { *!* - throw e; + throw err; */!* } @@ -44,4 +44,4 @@ function f() { f(); // cleanup! ``` -It's `finally` that guarantees the cleanup here. If we just put the code at the end of `f`, it wouldn't run. +It's `finally` that guarantees the cleanup here. If we just put the code at the end of `f`, it wouldn't run in these situations. diff --git a/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/task.md b/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/task.md index e84687343..b6dc81326 100644 --- a/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/task.md +++ b/1-js/10-error-handling/1-try-catch/1-finally-or-code-after/task.md @@ -6,12 +6,12 @@ importance: 5 Compare the two code fragments. -1. The first one uses `finally` to execute the code after `try..catch`: +1. The first one uses `finally` to execute the code after `try...catch`: ```js try { work work - } catch (e) { + } catch (err) { handle errors } finally { *!* @@ -19,12 +19,12 @@ Compare the two code fragments. */!* } ``` -2. The second fragment puts the cleaning right after `try..catch`: +2. The second fragment puts the cleaning right after `try...catch`: ```js try { work work - } catch (e) { + } catch (err) { handle errors } @@ -33,6 +33,6 @@ Compare the two code fragments. */!* ``` -We definitely need the cleanup after the work has started, doesn't matter if there was an error or not. +We definitely need the cleanup after the work, doesn't matter if there was an error or not. Is there an advantage here in using `finally` or both code fragments are equal? If there is such an advantage, then give an example when it matters. diff --git a/1-js/10-error-handling/1-try-catch/article.md b/1-js/10-error-handling/1-try-catch/article.md index d2a42dddc..cad2e1a3e 100644 --- a/1-js/10-error-handling/1-try-catch/article.md +++ b/1-js/10-error-handling/1-try-catch/article.md @@ -1,14 +1,14 @@ -# Error handling, "try..catch" +# Error handling, "try...catch" -No matter how great we are at programming, sometimes our scripts have errors. They may occur because of our mistakes, an unexpected user input, an erroneous server response and for a thousand of other reasons. +No matter how great we are at programming, sometimes our scripts have errors. They may occur because of our mistakes, an unexpected user input, an erroneous server response, and for a thousand other reasons. Usually, a script "dies" (immediately stops) in case of an error, printing it to console. -But there's a syntax construct `try..catch` that allows to "catch" errors and, instead of dying, do something more reasonable. +But there's a syntax construct `try...catch` that allows us to "catch" errors so the script can, instead of dying, do something more reasonable. -## The "try..catch" syntax +## The "try...catch" syntax -The `try..catch` construct has two main blocks: `try`, and then `catch`: +The `try...catch` construct has two main blocks: `try`, and then `catch`: ```js try { @@ -25,14 +25,14 @@ try { It works like this: 1. First, the code in `try {...}` is executed. -2. If there were no errors, then `catch(err)` is ignored: the execution reaches the end of `try` and then jumps over `catch`. -3. If an error occurs, then `try` execution is stopped, and the control flows to the beginning of `catch(err)`. The `err` variable (can use any name for it) contains an error object with details about what's happened. +2. If there were no errors, then `catch (err)` is ignored: the execution reaches the end of `try` and goes on, skipping `catch`. +3. If an error occurs, then the `try` execution is stopped, and control flows to the beginning of `catch (err)`. The `err` variable (we can use any name for it) will contain an error object with details about what happened.  -So, an error inside the `try {…}` block does not kill the script: we have a chance to handle it in `catch`. +So, an error inside the `try {...}` block does not kill the script -- we have a chance to handle it in `catch`. -Let's see more examples. +Let's look at some examples. - An errorless example: shows `alert` `(1)` and `(2)`: @@ -45,13 +45,11 @@ Let's see more examples. alert('End of try runs'); // *!*(2) <--*/!* - } catch(err) { + } catch (err) { alert('Catch is ignored, because there are no errors'); // (3) } - - alert("...Then the execution continues"); ``` - An example with an error: shows `(1)` and `(3)`: @@ -66,55 +64,53 @@ Let's see more examples. alert('End of try (never reached)'); // (2) - } catch(err) { + } catch (err) { alert(`Error has occurred!`); // *!*(3) <--*/!* } - - alert("...Then the execution continues"); ``` -````warn header="`try..catch` only works for runtime errors" -For `try..catch` to work, the code must be runnable. In other words, it should be valid JavaScript. +````warn header="`try...catch` only works for runtime errors" +For `try...catch` to work, the code must be runnable. In other words, it should be valid JavaScript. It won't work if the code is syntactically wrong, for instance it has unmatched curly braces: ```js run try { {{{{{{{{{{{{ -} catch(e) { +} catch (err) { alert("The engine can't understand this code, it's invalid"); } ``` -The JavaScript engine first reads the code, and then runs it. The errors that occur on the reading phrase are called "parse-time" errors and are unrecoverable (from inside that code). That's because the engine can't understand the code. +The JavaScript engine first reads the code, and then runs it. The errors that occur on the reading phase are called "parse-time" errors and are unrecoverable (from inside that code). That's because the engine can't understand the code. -So, `try..catch` can only handle errors that occur in the valid code. Such errors are called "runtime errors" or, sometimes, "exceptions". +So, `try...catch` can only handle errors that occur in valid code. Such errors are called "runtime errors" or, sometimes, "exceptions". ```` -````warn header="`try..catch` works synchronously" -If an exception happens in "scheduled" code, like in `setTimeout`, then `try..catch` won't catch it: +````warn header="`try...catch` works synchronously" +If an exception happens in "scheduled" code, like in `setTimeout`, then `try...catch` won't catch it: ```js run try { setTimeout(function() { noSuchVariable; // script will die here }, 1000); -} catch (e) { +} catch (err) { alert( "won't work" ); } ``` -That's because `try..catch` actually wraps the `setTimeout` call that schedules the function. But the function itself is executed later, when the engine has already left the `try..catch` construct. +That's because the function itself is executed later, when the engine has already left the `try...catch` construct. -To catch an exception inside a scheduled function, `try..catch` must be inside that function: +To catch an exception inside a scheduled function, `try...catch` must be inside that function: ```js run setTimeout(function() { try { - noSuchVariable; // try..catch handles the error! + noSuchVariable; // try...catch handles the error! } catch { alert( "error is caught here!" ); } @@ -129,15 +125,15 @@ When an error occurs, JavaScript generates an object containing the details abou ```js try { // ... -} catch(err) { // <-- the "error object", could use another word instead of err +} catch (err) { // <-- the "error object", could use another word instead of err // ... } ``` -For all built-in errors, the error object inside `catch` block has two main properties: +For all built-in errors, the error object has two main properties: `name` -: Error name. For an undefined variable that's `"ReferenceError"`. +: Error name. For instance, for an undefined variable that's `"ReferenceError"`. `message` : Textual message about error details. @@ -154,10 +150,10 @@ try { *!* lalala; // error, variable is not defined! */!* -} catch(err) { +} catch (err) { alert(err.name); // ReferenceError alert(err.message); // lalala is not defined - alert(err.stack); // ReferenceError: lalala is not defined at ... + alert(err.stack); // ReferenceError: lalala is not defined at (...call stack) // Can also show an error as a whole // The error is converted to string as "name: message" @@ -174,20 +170,20 @@ If we don't need error details, `catch` may omit it: ```js try { // ... -} catch { - // error object omitted +} catch { // <-- without (err) + // ... } ``` -## Using "try..catch" +## Using "try...catch" -Let's explore a real-life use case of `try..catch`. +Let's explore a real-life use case of `try...catch`. As we already know, JavaScript supports the [JSON.parse(str)](mdn:js/JSON/parse) method to read JSON-encoded values. Usually it's used to decode data received over the network, from the server or another source. -We receive it and call `JSON.parse`, like this: +We receive it and call `JSON.parse` like this: ```js run let json = '{"name":"John", "age": 30}'; // data from the server @@ -205,11 +201,11 @@ You can find more detailed information about JSON in the <info:json> chapter. **If `json` is malformed, `JSON.parse` generates an error, so the script "dies".** -Should we be satisfied with that? Of course, not! +Should we be satisfied with that? Of course not! This way, if something's wrong with the data, the visitor will never know that (unless they open the developer console). And people really don't like when something "just dies" without any error message. -Let's use `try..catch` to handle the error: +Let's use `try...catch` to handle the error: ```js run let json = "{ bad json }"; @@ -221,12 +217,12 @@ try { */!* alert( user.name ); // doesn't work -} catch (e) { +} catch (err) { *!* // ...the execution jumps here alert( "Our apologies, the data has errors, we'll try to request it one more time." ); - alert( e.name ); - alert( e.message ); + alert( err.name ); + alert( err.message ); */!* } ``` @@ -249,7 +245,7 @@ try { alert( user.name ); // no name! */!* -} catch (e) { +} catch (err) { alert( "doesn't execute" ); } ``` @@ -298,17 +294,17 @@ Let's see what kind of error `JSON.parse` generates: ```js run try { JSON.parse("{ bad json o_O }"); -} catch(e) { +} catch (err) { *!* - alert(e.name); // SyntaxError + alert(err.name); // SyntaxError */!* - alert(e.message); // Unexpected token o in JSON at position 0 + alert(err.message); // Unexpected token b in JSON at position 2 } ``` As we can see, that's a `SyntaxError`. -And in our case, the absence of `name` could be treated as a syntax error also, assuming that users must have a `name`. +And in our case, the absence of `name` is an error, as users must have a `name`. So let's throw it: @@ -327,8 +323,8 @@ try { alert( user.name ); -} catch(e) { - alert( "JSON Error: " + e.message ); // JSON Error: Incomplete data: no name +} catch (err) { + alert( "JSON Error: " + err.message ); // JSON Error: Incomplete data: no name } ``` @@ -338,9 +334,9 @@ Now `catch` became a single place for all error handling: both for `JSON.parse` ## Rethrowing -In the example above we use `try..catch` to handle incorrect data. But is it possible that *another unexpected error* occurs within the `try {...}` block? Like a variable is undefined or something else, not just that "incorrect data" thing. +In the example above we use `try...catch` to handle incorrect data. But is it possible that *another unexpected error* occurs within the `try {...}` block? Like a programming error (variable is not defined) or something else, not just this "incorrect data" thing. -Like this: +For example: ```js run let json = '{ "age": 30 }'; // incomplete data @@ -349,37 +345,41 @@ try { user = JSON.parse(json); // <-- forgot to put "let" before user // ... -} catch(err) { +} catch (err) { alert("JSON Error: " + err); // JSON Error: ReferenceError: user is not defined // (no JSON Error actually) } ``` -Of course, everything's possible! Programmers do make mistakes. Even in open-source utilities used by millions for decades -- suddenly a crazy bug may be discovered that leads to terrible hacks (like it happened with the `ssh` tool). +Of course, everything's possible! Programmers do make mistakes. Even in open-source utilities used by millions for decades -- suddenly a bug may be discovered that leads to terrible hacks. + +In our case, `try...catch` is placed to catch "incorrect data" errors. But by its nature, `catch` gets *all* errors from `try`. Here it gets an unexpected error, but still shows the same `"JSON Error"` message. That's wrong and also makes the code more difficult to debug. + +To avoid such problems, we can employ the "rethrowing" technique. The rule is simple: + +**Catch should only process errors that it knows and "rethrow" all others.** + +The "rethrowing" technique can be explained in more detail as: -In our case, `try..catch` is meant to catch "incorrect data" errors. But by its nature, `catch` gets *all* errors from `try`. Here it gets an unexpected error, but still shows the same `"JSON Error"` message. That's wrong and also makes the code more difficult to debug. +1. Catch gets all errors. +2. In the `catch (err) {...}` block we analyze the error object `err`. +3. If we don't know how to handle it, we do `throw err`. -Fortunately, we can find out which error we get, for instance from its `name`: +Usually, we can check the error type using the `instanceof` operator: ```js run try { user = { /*...*/ }; -} catch(e) { +} catch (err) { *!* - alert(e.name); // "ReferenceError" for accessing an undefined variable + if (err instanceof ReferenceError) { */!* + alert('ReferenceError'); // "ReferenceError" for accessing an undefined variable + } } ``` -The rule is simple: - -**Catch should only process errors that it knows and "rethrow" all others.** - -The "rethrowing" technique can be explained in more detail as: - -1. Catch gets all errors. -2. In `catch(err) {...}` block we analyze the error object `err`. -2. If we don't know how to handle it, then we do `throw err`. +We can also get the error class name from `err.name` property. All native errors have it. Another option is to read `err.constructor.name`. In the code below, we use rethrowing so that `catch` only handles `SyntaxError`: @@ -399,24 +399,24 @@ try { alert( user.name ); -} catch(e) { +} catch (err) { *!* - if (e.name == "SyntaxError") { - alert( "JSON Error: " + e.message ); + if (err instanceof SyntaxError) { + alert( "JSON Error: " + err.message ); } else { - throw e; // rethrow (*) + throw err; // rethrow (*) } */!* } ``` -The error throwing on line `(*)` from inside `catch` block "falls out" of `try..catch` and can be either caught by an outer `try..catch` construct (if it exists), or it kills the script. +The error throwing on line `(*)` from inside `catch` block "falls out" of `try...catch` and can be either caught by an outer `try...catch` construct (if it exists), or it kills the script. So the `catch` block actually handles only errors that it knows how to deal with and "skips" all others. -The example below demonstrates how such errors can be caught by one more level of `try..catch`: +The example below demonstrates how such errors can be caught by one more level of `try...catch`: ```js run function readData() { @@ -427,11 +427,11 @@ function readData() { *!* blabla(); // error! */!* - } catch (e) { + } catch (err) { // ... - if (e.name != 'SyntaxError') { + if (!(err instanceof SyntaxError)) { *!* - throw e; // rethrow (don't know how to deal with it) + throw err; // rethrow (don't know how to deal with it) */!* } } @@ -439,20 +439,20 @@ function readData() { try { readData(); -} catch (e) { +} catch (err) { *!* - alert( "External catch got: " + e ); // caught it! + alert( "External catch got: " + err ); // caught it! */!* } ``` -Here `readData` only knows how to handle `SyntaxError`, while the outer `try..catch` knows how to handle everything. +Here `readData` only knows how to handle `SyntaxError`, while the outer `try...catch` knows how to handle everything. -## try..catch..finally +## try...catch...finally Wait, that's not all. -The `try..catch` construct may have one more code clause: `finally`. +The `try...catch` construct may have one more code clause: `finally`. If it exists, it runs in all cases: @@ -464,7 +464,7 @@ The extended syntax looks like this: ```js *!*try*/!* { ... try to execute the code ... -} *!*catch*/!*(e) { +} *!*catch*/!* (err) { ... handle errors ... } *!*finally*/!* { ... execute always ... @@ -477,7 +477,7 @@ Try running this code: try { alert( 'try' ); if (confirm('Make an error?')) BAD_CODE(); -} catch (e) { +} catch (err) { alert( 'catch' ); } finally { alert( 'finally' ); @@ -489,7 +489,7 @@ The code has two ways of execution: 1. If you answer "Yes" to "Make an error?", then `try -> catch -> finally`. 2. If you say "No", then `try -> finally`. -The `finally` clause is often used when we start doing something before `try..catch` and want to finalize it in any case of outcome. +The `finally` clause is often used when we start doing something and want to finalize it in any case of outcome. For instance, we want to measure the time that a Fibonacci numbers function `fib(n)` takes. Naturally, we can start measuring before it runs and finish afterwards. But what if there's an error during the function call? In particular, the implementation of `fib(n)` in the code below returns an error for negative or non-integer numbers. @@ -513,7 +513,7 @@ let start = Date.now(); try { result = fib(num); -} catch (e) { +} catch (err) { result = 0; *!* } finally { @@ -521,24 +521,24 @@ try { } */!* -alert(result || "error occured"); +alert(result || "error occurred"); alert( `execution took ${diff}ms` ); ``` -You can check by running the code with entering `35` into `prompt` -- it executes normally, `finally` after `try`. And then enter `-1` -- there will be an immediate error, an the execution will take `0ms`. Both measurements are done correctly. +You can check by running the code with entering `35` into `prompt` -- it executes normally, `finally` after `try`. And then enter `-1` -- there will be an immediate error, and the execution will take `0ms`. Both measurements are done correctly. -In other words, there may be two ways to exit a function: either a `return` or `throw`. The `finally` clause handles them both. +In other words, the function may finish with `return` or `throw`, that doesn't matter. The `finally` clause executes in both cases. -```smart header="Variables are local inside `try..catch..finally`" -Please note that `result` and `diff` variables in the code above are declared *before* `try..catch`. +```smart header="Variables are local inside `try...catch...finally`" +Please note that `result` and `diff` variables in the code above are declared *before* `try...catch`. -Otherwise, if `let` were made inside the `{...}` block, it would only be visible inside of it. +Otherwise, if we declared `let` in `try` block, it would only be visible inside of it. ``` ````smart header="`finally` and `return`" -The `finally` clause works for *any* exit from `try..catch`. That includes an explicit `return`. +The `finally` clause works for *any* exit from `try...catch`. That includes an explicit `return`. In the example below, there's a `return` in `try`. In this case, `finally` is executed just before the control returns to the outer code. @@ -550,7 +550,7 @@ function func() { return 1; */!* - } catch (e) { + } catch (err) { /* ... */ } finally { *!* @@ -563,9 +563,9 @@ alert( func() ); // first works alert from finally, and then this one ``` ```` -````smart header="`try..finally`" +````smart header="`try...finally`" -The `try..finally` construct, without `catch` clause, is also useful. We apply it when we don't want to handle errors right here, but want to be sure that processes that we started are finalized. +The `try...finally` construct, without `catch` clause, is also useful. We apply it when we don't want to handle errors here (let them fall through), but want to be sure that processes that we started are finalized. ```js function func() { @@ -577,7 +577,7 @@ function func() { } } ``` -In the code above, an error inside `try` always falls out, because there's no `catch`. But `finally` works before the execution flow jumps outside. +In the code above, an error inside `try` always falls out, because there's no `catch`. But `finally` works before the execution flow leaves the function. ```` ## Global catch @@ -586,11 +586,11 @@ In the code above, an error inside `try` always falls out, because there's no `c The information from this section is not a part of the core JavaScript. ``` -Let's imagine we've got a fatal error outside of `try..catch`, and the script died. Like a programming error or something else terrible. +Let's imagine we've got a fatal error outside of `try...catch`, and the script died. Like a programming error or some other terrible thing. -Is there a way to react on such occurrences? We may want to log the error, show something to the user (normally they don't see error messages) etc. +Is there a way to react on such occurrences? We may want to log the error, show something to the user (normally they don't see error messages), etc. -There is none in the specification, but environments usually provide it, because it's really useful. For instance, Node.js has [process.on('uncaughtException')](https://nodejs.org/api/process.html#process_event_uncaughtexception) for that. And in the browser we can assign a function to special [window.onerror](mdn:api/GlobalEventHandlers/onerror) property. It will run in case of an uncaught error. +There is none in the specification, but environments usually provide it, because it's really useful. For instance, Node.js has [`process.on("uncaughtException")`](https://nodejs.org/api/process.html#process_event_uncaughtexception) for that. And in the browser we can assign a function to the special [window.onerror](mdn:api/GlobalEventHandlers/onerror) property, that will run in case of an uncaught error. The syntax: @@ -632,25 +632,25 @@ For instance: The role of the global handler `window.onerror` is usually not to recover the script execution -- that's probably impossible in case of programming errors, but to send the error message to developers. -There are also web-services that provide error-logging for such cases, like <https://errorception.com> or <http://www.muscula.com>. +There are also web-services that provide error-logging for such cases, like <https://muscula.com> or <https://www.sentry.io>. They work like this: 1. We register at the service and get a piece of JS (or a script URL) from them to insert on pages. -2. That JS script has a custom `window.onerror` function. +2. That JS script sets a custom `window.onerror` function. 3. When an error occurs, it sends a network request about it to the service. 4. We can log in to the service web interface and see errors. ## Summary -The `try..catch` construct allows to handle runtime errors. It literally allows to try running the code and catch errors that may occur in it. +The `try...catch` construct allows to handle runtime errors. It literally allows to "try" running the code and "catch" errors that may occur in it. The syntax is: ```js try { // run this code -} catch(err) { +} catch (err) { // if an error happened, then jump here // err is the error object } finally { @@ -658,18 +658,18 @@ try { } ``` -There may be no `catch` section or no `finally`, so `try..catch` and `try..finally` are also valid. +There may be no `catch` section or no `finally`, so shorter constructs `try...catch` and `try...finally` are also valid. Error objects have following properties: - `message` -- the human-readable error message. - `name` -- the string with error name (error constructor name). -- `stack` (non-standard) -- the stack at the moment of error creation. +- `stack` (non-standard, but well-supported) -- the stack at the moment of error creation. -If error is not needed, we can omit it by using `catch {` instead of `catch(err) {`. +If an error object is not needed, we can omit it by using `catch {` instead of `catch (err) {`. We can also generate our own errors using the `throw` operator. Technically, the argument of `throw` can be anything, but usually it's an error object inheriting from the built-in `Error` class. More on extending errors in the next chapter. -Rethrowing is a basic pattern of error handling: a `catch` block usually expects and knows how to handle the particular error type, so it should rethrow errors it doesn't know. +*Rethrowing* is a very important pattern of error handling: a `catch` block usually expects and knows how to handle the particular error type, so it should rethrow errors it doesn't know. -Even if we don't have `try..catch`, most environments allow to setup a "global" error handler to catch errors that "fall out". In-browser that's `window.onerror`. +Even if we don't have `try...catch`, most environments allow us to setup a "global" error handler to catch errors that "fall out". In-browser, that's `window.onerror`. diff --git a/1-js/10-error-handling/2-custom-errors/1-format-error/solution.md b/1-js/10-error-handling/2-custom-errors/1-format-error/solution.md index bb6b74cfa..754e68f9a 100644 --- a/1-js/10-error-handling/2-custom-errors/1-format-error/solution.md +++ b/1-js/10-error-handling/2-custom-errors/1-format-error/solution.md @@ -2,7 +2,7 @@ class FormatError extends SyntaxError { constructor(message) { super(message); - this.name = "FormatError"; + this.name = this.constructor.name; } } diff --git a/1-js/10-error-handling/2-custom-errors/article.md b/1-js/10-error-handling/2-custom-errors/article.md index 5079c746d..d28b07439 100644 --- a/1-js/10-error-handling/2-custom-errors/article.md +++ b/1-js/10-error-handling/2-custom-errors/article.md @@ -2,11 +2,11 @@ When we develop something, we often need our own error classes to reflect specific things that may go wrong in our tasks. For errors in network operations we may need `HttpError`, for database operations `DbError`, for searching operations `NotFoundError` and so on. -Our errors should support basic error properties like `message`, `name` and, preferably, `stack`. But they also may have other properties of their own, e.g. `HttpError` objects may have `statusCode` property with a value like `404` or `403` or `500`. +Our errors should support basic error properties like `message`, `name` and, preferably, `stack`. But they also may have other properties of their own, e.g. `HttpError` objects may have a `statusCode` property with a value like `404` or `403` or `500`. JavaScript allows to use `throw` with any argument, so technically our custom error classes don't need to inherit from `Error`. But if we inherit, then it becomes possible to use `obj instanceof Error` to identify error objects. So it's better to inherit from it. -As we build our application, our own errors naturally form a hierarchy, for instance `HttpTimeoutError` may inherit from `HttpError`, and so on. +As the application grows, our own errors naturally form a hierarchy. For instance, `HttpTimeoutError` may inherit from `HttpError`, and so on. ## Extending Error @@ -17,17 +17,13 @@ Here's an example of how a valid `json` may look: let json = `{ "name": "John", "age": 30 }`; ``` -Internally, we'll use `JSON.parse`. If it receives malformed `json`, then it throws `SyntaxError`. - -But even if `json` is syntactically correct, that doesn't mean that it's a valid user, right? It may miss the necessary data. For instance, it may not have `name` and `age` properties that are essential for our users. +Internally, we'll use `JSON.parse`. If it receives malformed `json`, then it throws `SyntaxError`. But even if `json` is syntactically correct, that doesn't mean that it's a valid user, right? It may miss the necessary data. For instance, it may not have `name` and `age` properties that are essential for our users. Our function `readUser(json)` will not only read JSON, but check ("validate") the data. If there are no required fields, or the format is wrong, then that's an error. And that's not a `SyntaxError`, because the data is syntactically correct, but another kind of error. We'll call it `ValidationError` and create a class for it. An error of that kind should also carry the information about the offending field. -Our `ValidationError` class should inherit from the built-in `Error` class. - -That class is built-in, but we should have its approximate code before our eyes, to understand what we're extending. +Our `ValidationError` class should inherit from the `Error` class. -So here you are: +The `Error` class is built-in, but here's its approximate code so we can understand what we're extending: ```js // The "pseudocode" for the built-in Error class defined by JavaScript itself @@ -35,14 +31,14 @@ class Error { constructor(message) { this.message = message; this.name = "Error"; // (different names for different built-in error classes) - this.stack = <nested calls>; // non-standard, but most environments support it + this.stack = <call stack>; // non-standard, but most environments support it } } ``` -Now let's go on and inherit `ValidationError` from it: +Now let's inherit `ValidationError` from it and try it in action: -```js run untrusted +```js run *!* class ValidationError extends Error { */!* @@ -65,10 +61,9 @@ try { } ``` -Please take a look at the constructor: +Please note: in the line `(1)` we call the parent constructor. JavaScript requires us to call `super` in the child constructor, so that's obligatory. The parent constructor sets the `message` property. -1. In the line `(1)` we call the parent constructor. JavaScript requires us to call `super` in the child constructor, so that's obligatory. The parent constructor sets the `message` property. -2. The parent constructor also sets the `name` property to `"Error"`, so in the line `(2)` we reset it to the right value. +The parent constructor also sets the `name` property to `"Error"`, so in the line `(2)` we reset it to the right value. Let's try to use it in `readUser(json)`: @@ -122,15 +117,15 @@ We could also look at `err.name`, like this: // instead of (err instanceof SyntaxError) } else if (err.name == "SyntaxError") { // (*) // ... -``` +``` The `instanceof` version is much better, because in the future we are going to extend `ValidationError`, make subtypes of it, like `PropertyRequiredError`. And `instanceof` check will continue to work for new inheriting classes. So that's future-proof. -Also it's important that if `catch` meets an unknown error, then it rethrows it in the line `(**)`. The `catch` only knows how to handle validation and syntax errors, other kinds (due to a typo in the code or such) should fall through. +Also it's important that if `catch` meets an unknown error, then it rethrows it in the line `(**)`. The `catch` block only knows how to handle validation and syntax errors, other kinds (caused by a typo in the code or other unknown reasons) should fall through. ## Further inheritance -The `ValidationError` class is very generic. Many things may go wrong. The property may be absent or it may be in a wrong format (like a string value for `age`). Let's make a more concrete class `PropertyRequiredError`, exactly for absent properties. It will carry additional information about the property that's missing. +The `ValidationError` class is very generic. Many things may go wrong. The property may be absent or it may be in a wrong format (like a string value for `age` instead of a number). Let's make a more concrete class `PropertyRequiredError`, exactly for absent properties. It will carry additional information about the property that's missing. ```js run class ValidationError extends Error { @@ -185,7 +180,7 @@ try { The new class `PropertyRequiredError` is easy to use: we only need to pass the property name: `new PropertyRequiredError(property)`. The human-readable `message` is generated by the constructor. -Please note that `this.name` in `PropertyRequiredError` constructor is again assigned manually. That may become a bit tedious -- to assign `this.name = <class name>` when creating each custom error. But there's a way out. We can make our own "basic error" class that removes this burden from our shoulders by using `this.constructor.name` for `this.name` in the constructor. And then inherit from it. +Please note that `this.name` in `PropertyRequiredError` constructor is again assigned manually. That may become a bit tedious -- to assign `this.name = <class name>` in every custom error class. We can avoid it by making our own "basic error" class that assigns `this.name = this.constructor.name`. And then inherit all our custom errors from it. Let's call it `MyError`. @@ -218,13 +213,41 @@ Now custom errors are much shorter, especially `ValidationError`, as we got rid ## Wrapping exceptions -The purpose of the function `readUser` in the code above is "to read the user data", right? There may occur different kinds of errors in the process. Right now we have `SyntaxError` and `ValidationError`, but in the future `readUser` function may grow: the new code will probably generate other kinds of errors. +The purpose of the function `readUser` in the code above is "to read the user data". There may occur different kinds of errors in the process. Right now we have `SyntaxError` and `ValidationError`, but in the future `readUser` function may grow and probably generate other kinds of errors. + +The code which calls `readUser` should handle these errors. Right now it uses multiple `if`s in the `catch` block, that check the class and handle known errors and rethrow the unknown ones. + +The scheme is like this: + +```js +try { + ... + readUser() // the potential error source + ... +} catch (err) { + if (err instanceof ValidationError) { + // handle validation errors + } else if (err instanceof SyntaxError) { + // handle syntax errors + } else { + throw err; // unknown error, rethrow it + } +} +``` + +In the code above we can see two types of errors, but there can be more. + +If the `readUser` function generates several kinds of errors, then we should ask ourselves: do we really want to check for all error types one-by-one every time? + +Often the answer is "No": we'd like to be "one level above all that". We just want to know if there was a "data reading error" -- why exactly it happened is often irrelevant (the error message describes it). Or, even better, we'd like to have a way to get the error details, but only if we need to. -The code which calls `readUser` should handle these errors. Right now it uses multiple `if` in the `catch` block to check for different error types and rethrow the unknown ones. But if `readUser` function generates several kinds of errors -- then we should ask ourselves: do we really want to check for all error types one-by-one in every code that calls `readUser`? +The technique that we describe here is called "wrapping exceptions". -Often the answer is "No": the outer code wants to be "one level above all that". It wants to have some kind of "data reading error". Why exactly it happened -- is often irrelevant (the error message describes it). Or, even better if there is a way to get error details, but only if we need to. +1. We'll make a new class `ReadError` to represent a generic "data reading" error. +2. The function `readUser` will catch data reading errors that occur inside it, such as `ValidationError` and `SyntaxError`, and generate a `ReadError` instead. +3. The `ReadError` object will keep the reference to the original error in its `cause` property. -So let's make a new class `ReadError` to represent such errors. If an error occurs inside `readUser`, we'll catch it there and generate `ReadError`. We'll also keep the reference to the original error in its `cause` property. Then the outer code will only have to check for `ReadError`. +Then the code that calls `readUser` will only have to check for `ReadError`, not for every kind of data reading errors. And if it needs more details of an error, it can check its `cause` property. Here's the code that defines `ReadError` and demonstrates its use in `readUser` and `try..catch`: @@ -296,12 +319,12 @@ try { In the code above, `readUser` works exactly as described -- catches syntax and validation errors and throws `ReadError` errors instead (unknown errors are rethrown as usual). -So the outer code checks `instanceof ReadError` and that's it. No need to list possible all error types. +So the outer code checks `instanceof ReadError` and that's it. No need to list all possible error types. -The approach is called "wrapping exceptions", because we take "low level exceptions" and "wrap" them into `ReadError` that is more abstract and more convenient to use for the calling code. It is widely used in object-oriented programming. +The approach is called "wrapping exceptions", because we take "low level" exceptions and "wrap" them into `ReadError` that is more abstract. It is widely used in object-oriented programming. ## Summary -- We can inherit from `Error` and other built-in error classes normally, just need to take care of `name` property and don't forget to call `super`. -- Most of the time, we should use `instanceof` to check for particular errors. It also works with inheritance. But sometimes we have an error object coming from the 3rd-party library and there's no easy way to get the class. Then `name` property can be used for such checks. -- Wrapping exceptions is a widespread technique when a function handles low-level exceptions and makes a higher-level object to report about the errors. Low-level exceptions sometimes become properties of that object like `err.cause` in the examples above, but that's not strictly required. +- We can inherit from `Error` and other built-in error classes normally. We just need to take care of the `name` property and don't forget to call `super`. +- We can use `instanceof` to check for particular errors. It also works with inheritance. But sometimes we have an error object coming from a 3rd-party library and there's no easy way to get its class. Then `name` property can be used for such checks. +- Wrapping exceptions is a widespread technique: a function handles low-level exceptions and creates higher-level errors instead of various low-level ones. Low-level exceptions sometimes become properties of that object like `err.cause` in the examples above, but that's not strictly required. diff --git a/1-js/11-async/01-callbacks/article.md b/1-js/11-async/01-callbacks/article.md index a9183c80f..18039db17 100644 --- a/1-js/11-async/01-callbacks/article.md +++ b/1-js/11-async/01-callbacks/article.md @@ -2,38 +2,53 @@ # Introduction: callbacks -Many actions in JavaScript are *asynchronous*. +```warn header="We use browser methods in examples here" +To demonstrate the use of callbacks, promises and other abstract concepts, we'll be using some browser methods: specifically, loading scripts and performing simple document manipulations. -For instance, take a look at the function `loadScript(src)`: +If you're not familiar with these methods, and their usage in the examples is confusing, you may want to read a few chapters from the [next part](/document) of the tutorial. + +Although, we'll try to make things clear anyway. There won't be anything really complex browser-wise. +``` + +Many functions are provided by JavaScript host environments that allow you to schedule *asynchronous* actions. In other words, actions that we initiate now, but they finish later. + +For instance, one such function is the `setTimeout` function. + +There are other real-world examples of asynchronous actions, e.g. loading scripts and modules (we'll cover them in later chapters). + +Take a look at the function `loadScript(src)`, that loads a script with the given `src`: ```js function loadScript(src) { + // creates a <script> tag and append it to the page + // this causes the script with given src to start loading and run when complete let script = document.createElement('script'); script.src = src; document.head.append(script); } ``` -The purpose of the function is to load a new script. When it adds the `<script src="…">` to the document, the browser loads and executes it. +It inserts into the document a new, dynamically created, tag `<script src="…">` with the given `src`. The browser automatically starts loading it and executes when complete. -We can use it like this: +We can use this function like this: ```js -// loads and executes the script +// load and execute the script at the given path loadScript('/my/script.js'); ``` -The function is called "asynchronously," because the action (script loading) finishes not now, but later. +The script is executed "asynchronously", as it starts loading now, but runs later, when the function has already finished. -The call initiates the script loading, then the execution continues. While the script is loading, the code below may finish executing, and if the loading takes time, other scripts may run meanwhile too. +If there's any code below `loadScript(…)`, it doesn't wait until the script loading finishes. ```js loadScript('/my/script.js'); -// the code below loadScript doesn't wait for the script loading to finish +// the code below loadScript +// doesn't wait for the script loading to finish // ... ``` -Now let's say we want to use the new script when it loads. It probably declares new functions, so we'd like to run them. +Let's say we need to use the new script as soon as it loads. It declares new functions, and we want to run them. But if we do that immediately after the `loadScript(…)` call, that wouldn't work: @@ -45,7 +60,7 @@ newFunction(); // no such function! */!* ``` -Naturally, the browser probably didn't have time to load the script. So the immediate call to the new function fails. As of now, the `loadScript` function doesn't provide a way to track the load completion. The script loads and eventually runs, that's all. But we'd like to know when it happens, to use new functions and variables from that script. +Naturally, the browser probably didn't have time to load the script. As of now, the `loadScript` function doesn't provide a way to track the load completion. The script loads and eventually runs, that's all. But we'd like to know when it happens, to use new functions and variables from that script. Let's add a `callback` function as a second argument to `loadScript` that should execute when the script loads: @@ -62,6 +77,8 @@ function loadScript(src, *!*callback*/!*) { } ``` +The `onload` event is described in the article <info:onload-onerror#loading-a-script>, it basically executes a function after the script is loaded and executed. + Now if we want to call new functions from the script, we should write that in the callback: ```js @@ -86,15 +103,15 @@ function loadScript(src, callback) { *!* loadScript('https://cdnjs.cloudflare.com/ajax/libs/lodash.js/3.2.0/lodash.js', script => { - alert(`Cool, the ${script.src} is loaded`); - alert( _ ); // function declared in the loaded script + alert(`Cool, the script ${script.src} is loaded`); + alert( _ ); // _ is a function declared in the loaded script }); */!* ``` That's called a "callback-based" style of asynchronous programming. A function that does something asynchronously should provide a `callback` argument where we put the function to run after it's complete. -Here we did it in `loadScript`, but of course, it's a general approach. +Here we did it in `loadScript`, but of course it's a general approach. ## Callback in callback @@ -131,7 +148,7 @@ loadScript('/my/script.js', function(script) { }); */!* - }) + }); }); ``` @@ -144,7 +161,7 @@ In the above examples we didn't consider errors. What if the script loading fail Here's an improved version of `loadScript` that tracks loading errors: -```js run +```js function loadScript(src, callback) { let script = document.createElement('script'); script.src = src; @@ -181,9 +198,9 @@ So the single `callback` function is used both for reporting errors and passing ## Pyramid of Doom -From the first look, it's a viable way of asynchronous coding. And indeed it is. For one or maybe two nested calls it looks fine. +At first glance, it looks like a viable approach to asynchronous coding. And indeed it is. For one or maybe two nested calls it looks fine. -But for multiple asynchronous actions that follow one after another we'll have code like this: +But for multiple asynchronous actions that follow one after another, we'll have code like this: ```js loadScript('1.js', function(error, script) { @@ -208,20 +225,47 @@ loadScript('1.js', function(error, script) { }); } - }) + }); } }); ``` In the code above: -1. We load `1.js`, then if there's no error. -2. We load `2.js`, then if there's no error. +1. We load `1.js`, then if there's no error... +2. We load `2.js`, then if there's no error... 3. We load `3.js`, then if there's no error -- do something else `(*)`. -As calls become more nested, the code becomes deeper and increasingly more difficult to manage, especially if we have a real code instead of `...`, that may include more loops, conditional statements and so on. +As calls become more nested, the code becomes deeper and increasingly more difficult to manage, especially if we have real code instead of `...` that may include more loops, conditional statements and so on. That's sometimes called "callback hell" or "pyramid of doom." +<<<<<<< HEAD +======= +<!-- +loadScript('1.js', function(error, script) { + if (error) { + handleError(error); + } else { + // ... + loadScript('2.js', function(error, script) { + if (error) { + handleError(error); + } else { + // ... + loadScript('3.js', function(error, script) { + if (error) { + handleError(error); + } else { + // ... + } + }); + } + }); + } +}); +--> + +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  The "pyramid" of nested calls grows to the right with every asynchronous action. Soon it spirals out of control. @@ -257,15 +301,15 @@ function step3(error, script) { } else { // ...continue after all scripts are loaded (*) } -}; +} ``` -See? It does the same, and there's no deep nesting now because we made every action a separate top-level function. +See? It does the same thing, and there's no deep nesting now because we made every action a separate top-level function. It works, but the code looks like a torn apart spreadsheet. It's difficult to read, and you probably noticed that one needs to eye-jump between pieces while reading it. That's inconvenient, especially if the reader is not familiar with the code and doesn't know where to eye-jump. -Also, the functions named `step*` are all of single use, they are created only to avoid the "pyramid of doom." No one is going to reuse them outside of the action chain. So there's a bit of a namespace cluttering here. +Also, the functions named `step*` are all of single use, they are created only to avoid the "pyramid of doom." No one is going to reuse them outside of the action chain. So there's a bit of namespace cluttering here. We'd like to have something better. -Luckily, there are other ways to avoid such pyramids. One of the best ways is to use "promises," described in the next chapter. +Luckily, there are other ways to avoid such pyramids. One of the best ways is to use "promises", described in the next chapter. diff --git a/1-js/11-async/02-promise-basics/03-animate-circle-promise/solution.view/index.html b/1-js/11-async/02-promise-basics/03-animate-circle-promise/solution.view/index.html index 3229daf89..6052f009e 100644 --- a/1-js/11-async/02-promise-basics/03-animate-circle-promise/solution.view/index.html +++ b/1-js/11-async/02-promise-basics/03-animate-circle-promise/solution.view/index.html @@ -10,7 +10,7 @@ text-align: center; } .circle { - transition-property: width, height, margin-left, margin-top; + transition-property: width, height; transition-duration: 2s; position: fixed; transform: translateX(-50%) translateY(-50%); diff --git a/1-js/11-async/02-promise-basics/article.md b/1-js/11-async/02-promise-basics/article.md index 7e05a2ee8..232dbadae 100644 --- a/1-js/11-async/02-promise-basics/article.md +++ b/1-js/11-async/02-promise-basics/article.md @@ -1,14 +1,14 @@ # Promise -Imagine that you're a top singer, and fans ask day and night for your upcoming single. +Imagine that you're a top singer, and fans ask day and night for your upcoming song. -To get some relief, you promise to send it to them when it's published. You give your fans a list to which they can subscribe for updates. They can fill in their email addresses, so that when the song becomes available, all subscribed parties instantly receive it. And even if something goes very wrong, say, if plans to publish the song are cancelled, they will still be notified. +To get some relief, you promise to send it to them when it's published. You give your fans a list. They can fill in their email addresses, so that when the song becomes available, all subscribed parties instantly receive it. And even if something goes very wrong, say, a fire in the studio, so that you can't publish the song, they will still be notified. -Everyone is happy, because the people don't crowd you any more, and fans, because they won't miss the single. +Everyone is happy: you, because the people don't crowd you anymore, and fans, because they won't miss the song. This is a real-life analogy for things we often have in programming: -1. A "producing code" that does something and takes time. For instance, the code loads a remote script. That's a "singer". +1. A "producing code" that does something and takes time. For instance, some code that loads the data over a network. That's a "singer". 2. A "consuming code" that wants the result of the "producing code" once it's ready. Many functions may need that result. These are the "fans". 3. A *promise* is a special JavaScript object that links the "producing code" and the "consuming code" together. In terms of our analogy: this is the "subscription list". The "producing code" takes whatever time it needs to produce the promised result, and the "promise" makes that result available to all of the subscribed code when it's ready. @@ -22,29 +22,35 @@ let promise = new Promise(function(resolve, reject) { }); ``` -The function passed to `new Promise` is called the *executor*. When the promise is created, this executor function runs automatically. It contains the producing code, that should eventually produce a result. In terms of the analogy above: the executor is the "singer". +The function passed to `new Promise` is called the *executor*. When `new Promise` is created, the executor runs automatically. It contains the producing code which should eventually produce the result. In terms of the analogy above: the executor is the "singer". -The resulting `promise` object has internal properties: +Its arguments `resolve` and `reject` are callbacks provided by JavaScript itself. Our code is only inside the executor. -- `state` — initially "pending", then changes to either "fulfilled" or "rejected", -- `result` — an arbitrary value of your choosing, initially `undefined`. +When the executor obtains the result, be it soon or late, doesn't matter, it should call one of these callbacks: -When the executor finishes the job, it should call one of the functions that it gets as arguments: +- `resolve(value)` — if the job is finished successfully, with result `value`. +- `reject(error)` — if an error has occurred, `error` is the error object. -- `resolve(value)` — to indicate that the job finished successfully: - - sets `state` to `"fulfilled"`, - - sets `result` to `value`. -- `reject(error)` — to indicate that an error occurred: - - sets `state` to `"rejected"`, - - sets `result` to `error`. +So to summarize: the executor runs automatically and attempts to perform a job. When it is finished with the attempt, it calls `resolve` if it was successful or `reject` if there was an error. +<<<<<<< HEAD  +======= +The `promise` object returned by the `new Promise` constructor has these internal properties: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -Later we'll see how these changes become known to "fans". +- `state` — initially `"pending"`, then changes to either `"fulfilled"` when `resolve` is called or `"rejected"` when `reject` is called. +- `result` — initially `undefined`, then changes to `value` when `resolve(value)` is called or `error` when `reject(error)` is called. -Here's an example of a Promise constructor and a simple executor function with its "producing code" (the `setTimeout`): +So the executor eventually moves `promise` to one of these states: -```js run + + +Later we'll see how "fans" can subscribe to these changes. + +Here's an example of a promise constructor and a simple executor function with "producing code" that takes time (via `setTimeout`): + +```js let promise = new Promise(function(resolve, reject) { // the function is executed automatically when the promise is constructed @@ -55,12 +61,16 @@ let promise = new Promise(function(resolve, reject) { We can see two things by running the code above: -1. The executor is called automatically and immediately (by the `new Promise`). -2. The executor receives two arguments: `resolve` and `reject` — these functions are pre-defined by the JavaScript engine. So we don't need to create them. Instead, we should write the executor to call them when ready. +1. The executor is called automatically and immediately (by `new Promise`). +2. The executor receives two arguments: `resolve` and `reject`. These functions are pre-defined by the JavaScript engine, so we don't need to create them. We should only call one of them when ready. -After one second of "processing" the executor calls `resolve("done")` to produce the result: + After one second of "processing", the executor calls `resolve("done")` to produce the result. This changes the state of the `promise` object: +<<<<<<< HEAD  +======= +  +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b That was an example of a successful job completion, a "fulfilled promise". @@ -73,20 +83,28 @@ let promise = new Promise(function(resolve, reject) { }); ``` +<<<<<<< HEAD  +======= +The call to `reject(...)` moves the promise object to `"rejected"` state: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -To summarize, the executor should do a job (something that takes time usually) and then call `resolve` or `reject` to change the state of the corresponding Promise object. + -The Promise that is either resolved or rejected is called "settled", as opposed to a "pending" Promise. +To summarize, the executor should perform a job (usually something that takes time) and then call `resolve` or `reject` to change the state of the corresponding promise object. + +A promise that is either resolved or rejected is called "settled", as opposed to an initially "pending" promise. ````smart header="There can be only a single result or an error" -The executor should call only one `resolve` or one `reject`. The promise's state change is final. +The executor should call only one `resolve` or one `reject`. Any state change is final. All further calls of `resolve` and `reject` are ignored: ```js let promise = new Promise(function(resolve, reject) { +*!* resolve("done"); +*/!* reject(new Error("…")); // ignored setTimeout(() => resolve("…")); // ignored @@ -99,7 +117,7 @@ Also, `resolve`/`reject` expect only one argument (or none) and will ignore addi ```` ```smart header="Reject with `Error` objects" -In case something goes wrong, we can call `reject` with any type of argument (just like `resolve`). But it is recommended to use `Error` objects (or objects that inherit from `Error`). The reasoning for that will soon become apparent. +In case something goes wrong, the executor should call `reject`. That can be done with any type of argument (just like `resolve`). But it is recommended to use `Error` objects (or objects that inherit from `Error`). The reasoning for that will soon become apparent. ``` ````smart header="Immediately calling `resolve`/`reject`" @@ -112,18 +130,18 @@ let promise = new Promise(function(resolve, reject) { }); ``` -For instance, this might happen when we start to do a job but then see that everything has already been completed. +For instance, this might happen when we start to do a job but then see that everything has already been completed and cached. -That's fine. We immediately have a resolved Promise, nothing wrong with that. +That's fine. We immediately have a resolved promise. ```` ```smart header="The `state` and `result` are internal" -The properties `state` and `result` of the Promise object are internal. We can't directly access them from our "consuming code". We can use the methods `.then`/`.catch`/`.finally` for that. They are described below. +The properties `state` and `result` of the Promise object are internal. We can't directly access them. We can use the methods `.then`/`.catch`/`.finally` for that. They are described below. ``` -## Consumers: then, catch, finally +## Consumers: then, catch -A Promise object serves as a link between the executor (the "producing code" or "singer") and the consuming functions (the "fans"), which will receive the result or error. Consuming functions can be registered (subscribed) using methods `.then`, `.catch` and `.finally`. +A Promise object serves as a link between the executor (the "producing code" or "singer") and the consuming functions (the "fans"), which will receive the result or error. Consuming functions can be registered (subscribed) using the methods `.then` and `.catch`. ### then @@ -138,15 +156,9 @@ promise.then( ); ``` -The first argument of `.then` is a function that: +The first argument of `.then` is a function that runs when the promise is resolved and receives the result. -1. runs when the Promise is resolved, and -2. receives the result. - -The second argument of `.then` is a function that: - -1. runs when the Promise is rejected, and -2. receives the error. +The second argument of `.then` is a function that runs when the promise is rejected and receives the error. For instance, here's a reaction to a successfully resolved promise: @@ -166,7 +178,7 @@ promise.then( The first function was executed. -And in the case of a rejection -- the second one: +And in the case of a rejection, the second one: ```js run let promise = new Promise(function(resolve, reject) { @@ -212,73 +224,99 @@ promise.catch(alert); // shows "Error: Whoops!" after 1 second The call `.catch(f)` is a complete analog of `.then(null, f)`, it's just a shorthand. -### finally +## Cleanup: finally Just like there's a `finally` clause in a regular `try {...} catch {...}`, there's `finally` in promises. -The call `.finally(f)` is similar to `.then(f, f)` in the sense that it always runs when the promise is settled: be it resolve or reject. +The call `.finally(f)` is similar to `.then(f, f)` in the sense that `f` runs always, when the promise is settled: be it resolve or reject. + +The idea of `finally` is to set up a handler for performing cleanup/finalizing after the previous operations are complete. -`finally` is a good handler for performing cleanup, e.g. stopping our loading indicators, as they are not needed any more, no matter what the outcome is. +E.g. stopping loading indicators, closing no longer needed connections, etc. -Like this: +Think of it as a party finisher. Irresepective of whether a party was good or bad, how many friends were in it, we still need (or at least should) do a cleanup after it. + +The code may look like this: ```js new Promise((resolve, reject) => { - /* do something that takes time, and then call resolve/reject */ + /* do something that takes time, and then call resolve or maybe reject */ }) *!* // runs when the promise is settled, doesn't matter successfully or not .finally(() => stop loading indicator) + // so the loading indicator is always stopped before we go on */!* .then(result => show result, err => show error) ``` -It's not exactly an alias though. There are several important differences: +Please note that `finally(f)` isn't exactly an alias of `then(f,f)` though. + +There are important differences: 1. A `finally` handler has no arguments. In `finally` we don't know whether the promise is successful or not. That's all right, as our task is usually to perform "general" finalizing procedures. -2. Finally passes through results and errors to the next handler. + + Please take a look at the example above: as you can see, the `finally` handler has no arguments, and the promise outcome is handled by the next handler. +2. A `finally` handler "passes through" the result or error to the next suitable handler. For instance, here the result is passed through `finally` to `then`: + ```js run new Promise((resolve, reject) => { - setTimeout(() => resolve("result"), 2000) + setTimeout(() => resolve("value"), 2000); }) - .finally(() => alert("Promise ready")) - .then(result => alert(result)); // <-- .then handles the result + .finally(() => alert("Promise ready")) // triggers first + .then(result => alert(result)); // <-- .then shows "value" ``` - And here there's an error in the promise, passed through `finally` to `catch`: + As you can see, the `value` returned by the first promise is passed through `finally` to the next `then`. + + That's very convenient, because `finally` is not meant to process a promise result. As said, it's a place to do generic cleanup, no matter what the outcome was. + + And here's an example of an error, for us to see how it's passed through `finally` to `catch`: ```js run new Promise((resolve, reject) => { throw new Error("error"); }) - .finally(() => alert("Promise ready")) - .catch(err => alert(err)); // <-- .catch handles the error object - ``` + .finally(() => alert("Promise ready")) // triggers first + .catch(err => alert(err)); // <-- .catch shows the error + ``` + +3. A `finally` handler also shouldn't return anything. If it does, the returned value is silently ignored. + + The only exception to this rule is when a `finally` handler throws an error. Then this error goes to the next handler, instead of any previous outcome. - That's very convenient, because finally is not meant to process promise results. So it passes them through. +To summarize: - We'll talk more about promise chaining and result-passing between handlers in the next chapter. +- A `finally` handler doesn't get the outcome of the previous handler (it has no arguments). This outcome is passed through instead, to the next suitable handler. +- If a `finally` handler returns something, it's ignored. +- When `finally` throws an error, then the execution goes to the nearest error handler. -3. Last, but not least, `.finally(f)` is a more convenient syntax than `.then(f, f)`: no need to duplicate the function. +These features are helpful and make things work just the right way if we use `finally` how it's supposed to be used: for generic cleanup procedures. -````smart header="On settled promises handlers runs immediately" -If a promise is pending, `.then/catch/finally` handlers wait for the result. Otherwise, if a promise has already settled, they execute immediately: +````smart header="We can attach handlers to settled promises" +If a promise is pending, `.then/catch/finally` handlers wait for its outcome. + +Sometimes, it might be that a promise is already settled when we add a handler to it. + +In such case, these handlers just run immediately: ```js run -// an immediately resolved promise +// the promise becomes resolved immediately upon creation let promise = new Promise(resolve => resolve("done!")); promise.then(alert); // done! (shows up right now) ``` -The good thing is: a `.then` handler is guaranteed to run whether the promise takes time or settles it immediately. +Note that this makes promises more powerful than the real life "subscription list" scenario. If the singer has already released their song and then a person signs up on the subscription list, they probably won't receive that song. Subscriptions in real life must be done prior to the event. + +Promises are more flexible. We can add handlers any time: if the result is already there, they just execute. ```` -Next, let's see more practical examples of how promises can help us to write asynchronous code. +## Example: loadScript [#loadscript] -## Example: loadScript +Next, let's see more practical examples of how promises can help us write asynchronous code. We've got the `loadScript` function for loading a script from the previous chapter. @@ -301,7 +339,7 @@ Let's rewrite it using Promises. The new function `loadScript` will not require a callback. Instead, it will create and return a Promise object that resolves when the loading is complete. The outer code can add handlers (subscribing functions) to it using `.then`: ```js run -function loadScript(src) { +function loadScript(src) { return new Promise(function(resolve, reject) { let script = document.createElement('script'); script.src = src; @@ -324,7 +362,7 @@ promise.then( error => alert(`Error: ${error.message}`) ); -promise.then(script => alert('One more handler to do something else!')); +promise.then(script => alert('Another handler...')); ``` We can immediately see a few benefits over the callback-based pattern: @@ -335,4 +373,4 @@ We can immediately see a few benefits over the callback-based pattern: | Promises allow us to do things in the natural order. First, we run `loadScript(script)`, and `.then` we write what to do with the result. | We must have a `callback` function at our disposal when calling `loadScript(script, callback)`. In other words, we must know what to do with the result *before* `loadScript` is called. | | We can call `.then` on a Promise as many times as we want. Each time, we're adding a new "fan", a new subscribing function, to the "subscription list". More about this in the next chapter: [](info:promise-chaining). | There can be only one callback. | -So Promises give us better code flow and flexibility. But there's more. We'll see that in the next chapters. +So promises give us better code flow and flexibility. But there's more. We'll see that in the next chapters. diff --git a/1-js/11-async/02-promise-basics/promise-reject-1.svg b/1-js/11-async/02-promise-basics/promise-reject-1.svg index 1ed00b191..73601c5f9 100644 --- a/1-js/11-async/02-promise-basics/promise-reject-1.svg +++ b/1-js/11-async/02-promise-basics/promise-reject-1.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="535" height="106" viewBox="0 0 535 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-reject-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M184 34v58H9V34h175z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="10" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="21" y="54">state: "pending"</tspan> <tspan x="21" y="74">result: undefined</tspan></text><text id="reject(error)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="212.4" y="49">reject(error)</tspan></text><path id="Rectangle-1-Copy-3" fill="#FEF1F0" stroke="#D35155" stroke-width="2" d="M353 34h175v58H353z"/><text id="state:-"rejected"-re" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "rejected"</tspan> <tspan x="368" y="75">result: error</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="535" height="106" viewBox="0 0 535 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-reject-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M184 34v58H9V34h175z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="10" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="21" y="54">state: "pending"</tspan> <tspan x="21" y="74">result: undefined</tspan></text><text id="reject(error)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="212.4" y="49">reject(error)</tspan></text><path id="Rectangle-1-Copy-3" fill="#FEF1F0" stroke="#D35155" stroke-width="2" d="M353 34h175v58H353z"/><text id="state:-"rejected"-re" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "rejected"</tspan> <tspan x="368" y="75">result: error</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="535" height="106" viewBox="0 0 535 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-reject-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M184 34v58H9V34h175z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="10" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="21" y="54">state: "pending"</tspan> <tspan x="21" y="74">result: undefined</tspan></text><text id="reject(error)" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="212.4" y="49">reject(error)</tspan></text><path id="Rectangle-1-Copy-3" fill="#FEF1F0" stroke="#D35155" stroke-width="2" d="M353 34h175v58H353z"/><text id="state:-"rejected"-re" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "rejected"</tspan> <tspan x="368" y="75">result: error</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z" transform="matrix(1 0 0 -1 0 120)"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/11-async/02-promise-basics/promise-resolve-1.svg b/1-js/11-async/02-promise-basics/promise-resolve-1.svg index bb0b918db..8d95ce794 100644 --- a/1-js/11-async/02-promise-basics/promise-resolve-1.svg +++ b/1-js/11-async/02-promise-basics/promise-resolve-1.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="538" height="106" viewBox="0 0 538 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-resolve-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M187 34v58H6V34h181z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="19" y="54">state: "pending"</tspan> <tspan x="19" y="74">result: undefined</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z"/><text id="resolve("done")" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="200.5" y="49">resolve("done")</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#478964" stroke-width="2" d="M353 35h181v57H353z"/><text id="state:-"fulfilled"-r" fill="#478964" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "fulfilled"</tspan> <tspan x="368" y="75">result: "done"</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="538" height="106" viewBox="0 0 538 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-resolve-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M187 34v58H6V34h181z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="19" y="54">state: "pending"</tspan> <tspan x="19" y="74">result: undefined</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z"/><text id="resolve("done")" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="200.5" y="49">resolve("done")</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#478964" stroke-width="2" d="M353 35h181v57H353z"/><text id="state:-"fulfilled"-r" fill="#478964" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "fulfilled"</tspan> <tspan x="368" y="75">result: "done"</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="538" height="106" viewBox="0 0 538 106"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="promise-resolve-1.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M187 34v58H6V34h181z"/><text id="new-Promise(executor" fill="#7E7C7B" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="8" y="22">new Promise(executor)</tspan></text><text id="state:-"pending"-res" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="19" y="54">state: "pending"</tspan> <tspan x="19" y="74">result: undefined</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M329 53l14 7-14 7v-6H196v-2h133v-6z" transform="matrix(1 0 0 -1 0 120)"/><text id="resolve("done")" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="200.5" y="49">resolve("done")</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#478964" stroke-width="2" d="M353 35h181v57H353z"/><text id="state:-"fulfilled"-r" fill="#478964" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="368" y="55">state: "fulfilled"</tspan> <tspan x="368" y="75">result: "done"</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/11-async/03-promise-chaining/01-then-vs-catch/solution.md b/1-js/11-async/03-promise-chaining/01-then-vs-catch/solution.md index 5a00f348f..bdd1c643b 100644 --- a/1-js/11-async/03-promise-chaining/01-then-vs-catch/solution.md +++ b/1-js/11-async/03-promise-chaining/01-then-vs-catch/solution.md @@ -1,4 +1,4 @@ -The short answer is: **no, they are not the equal**: +The short answer is: **no, they are not equal**: The difference is that if an error happens in `f1`, then it is handled by `.catch` here: @@ -17,4 +17,4 @@ promise That's because an error is passed down the chain, and in the second code piece there's no chain below `f1`. -In other words, `.then` passes results/errors to the next `.then/catch`. So in the first example, there's a `catch` below, and in the second one -- there isn't, so the error is unhandled. +In other words, `.then` passes results/errors to the next `.then/catch`. So in the first example, there's a `catch` below, and in the second one there isn't, so the error is unhandled. diff --git a/1-js/11-async/03-promise-chaining/article.md b/1-js/11-async/03-promise-chaining/article.md index 0747842f1..a33ca258c 100644 --- a/1-js/11-async/03-promise-chaining/article.md +++ b/1-js/11-async/03-promise-chaining/article.md @@ -1,7 +1,7 @@ # Promises chaining -Let's return to the problem mentioned in the chapter <info:callbacks>: we have a sequence of asynchronous tasks to be done one after another. For instance, loading scripts. How can we code it well? +Let's return to the problem mentioned in the chapter <info:callbacks>: we have a sequence of asynchronous tasks to be performed one after another — for instance, loading scripts. How can we code it well? Promises provide a couple of recipes to do that. @@ -36,36 +36,18 @@ The idea is that the result is passed through the chain of `.then` handlers. Here the flow is: 1. The initial promise resolves in 1 second `(*)`, -2. Then the `.then` handler is called `(**)`. -3. The value that it returns is passed to the next `.then` handler `(***)` +2. Then the `.then` handler is called `(**)`, which in turn creates a new promise (resolved with `2` value). +3. The next `then` `(***)` gets the result of the previous one, processes it (doubles) and passes it to the next handler. 4. ...and so on. As the result is passed along the chain of handlers, we can see a sequence of `alert` calls: `1` -> `2` -> `4`.  -The whole thing works, because a call to `promise.then` returns a promise, so that we can call the next `.then` on it. +The whole thing works, because every call to a `.then` returns a new promise, so that we can call the next `.then` on it. When a handler returns a value, it becomes the result of that promise, so the next `.then` is called with it. -To make these words more clear, here's the start of the chain: - -```js run -new Promise(function(resolve, reject) { - - setTimeout(() => resolve(1), 1000); - -}).then(function(result) { - - alert(result); - return result * 2; // <-- (1) - -}) // <-- (2) -// .then… -``` - -The value returned by `.then` is a promise, that's why we are able to add another `.then` at `(2)`. When the value is returned in `(1)`, that promise becomes resolved, so the next handler runs with the value. - **A classic newbie error: technically we can also add many `.then` to a single promise. This is not chaining.** For example: @@ -90,7 +72,7 @@ promise.then(function(result) { }); ``` -What we did here is just several handlers to one promise. They don't pass the result to each other, instead they process it independently. +What we did here is just adding several handlers to one promise. They don't pass the result to each other; instead they process it independently. Here's the picture (compare it with the chaining above): @@ -102,9 +84,9 @@ In practice we rarely need multiple handlers for one promise. Chaining is used m ## Returning promises -Normally, a value returned by a `.then` handler is immediately passed to the next handler. But there's an exception. +A handler, used in `.then(handler)` may create and return a promise. -If the returned value is a promise, then the further execution is suspended until it settles. After that, the result of that promise is given to the next `.then` handler. +In that case further handlers wait until it settles, and then get its result. For instance: @@ -138,15 +120,15 @@ new Promise(function(resolve, reject) { }); ``` -Here the first `.then` shows `1` returns `new Promise(…)` in the line `(*)`. After one second it resolves, and the result (the argument of `resolve`, here it's `result*2`) is passed on to handler of the second `.then` in the line `(**)`. It shows `2` and does the same thing. +Here the first `.then` shows `1` and returns `new Promise(…)` in the line `(*)`. After one second it resolves, and the result (the argument of `resolve`, here it's `result * 2`) is passed on to the handler of the second `.then`. That handler is in the line `(**)`, it shows `2` and does the same thing. -So the output is again 1 -> 2 -> 4, but now with 1 second delay between `alert` calls. +So the output is the same as in the previous example: 1 -> 2 -> 4, but now with 1 second delay between `alert` calls. Returning promises allows us to build chains of asynchronous actions. ## Example: loadScript -Let's use this feature with `loadScript` to load scripts one by one, in sequence: +Let's use this feature with the promisified `loadScript`, defined in the [previous chapter](info:promise-basics#loadscript), to load scripts one by one, in sequence: ```js run loadScript("/article/promise-chaining/one.js") @@ -182,9 +164,9 @@ loadScript("/article/promise-chaining/one.js") Here each `loadScript` call returns a promise, and the next `.then` runs when it resolves. Then it initiates the loading of the next script. So scripts are loaded one after another. -We can add more asynchronous actions to the chain. Please note that code is still "flat", it grows down, not to the right. There are no signs of "pyramid of doom". +We can add more asynchronous actions to the chain. Please note that the code is still "flat" — it grows down, not to the right. There are no signs of the "pyramid of doom". -Please note that technically we can add `.then` directly to each `loadScript`, like this: +Technically, we could add `.then` directly to each `loadScript`, like this: ```js run loadScript("/article/promise-chaining/one.js").then(script1 => { @@ -207,11 +189,9 @@ Sometimes it's ok to write `.then` directly, because the nested function has acc ````smart header="Thenables" -To be precise, `.then` may return an arbitrary "thenable" object, and it will be treated the same way as a promise. +To be precise, a handler may return not exactly a promise, but a so-called "thenable" object - an arbitrary object that has a method `.then`. It will be treated the same way as a promise. -A "thenable" object is any object with a method `.then`. - -The idea is that 3rd-party libraries may implement "promise-compatible" objects of their own. They can have extended set of methods, but also be compatible with native promises, because they implement `.then`. +The idea is that 3rd-party libraries may implement "promise-compatible" objects of their own. They can have an extended set of methods, but also be compatible with native promises, because they implement `.then`. Here's an example of a thenable object: @@ -229,22 +209,24 @@ class Thenable { new Promise(resolve => resolve(1)) .then(result => { +*!* return new Thenable(result); // (*) +*/!* }) .then(alert); // shows 2 after 1000ms ``` -JavaScript checks the object returned by `.then` handler in the line `(*)`: if it has a callable method named `then`, then it calls that method providing native functions `resolve`, `reject` as arguments (similar to executor) and waits until one of them is called. In the example above `resolve(2)` is called after 1 second `(**)`. Then the result is passed further down the chain. +JavaScript checks the object returned by the `.then` handler in line `(*)`: if it has a callable method named `then`, then it calls that method providing native functions `resolve`, `reject` as arguments (similar to an executor) and waits until one of them is called. In the example above `resolve(2)` is called after 1 second `(**)`. Then the result is passed further down the chain. -This feature allows to integrate custom objects with promise chains without having to inherit from `Promise`. +This feature allows us to integrate custom objects with promise chains without having to inherit from `Promise`. ```` ## Bigger example: fetch -In frontend programming promises are often used for network requests. So let's see an extended example of that. +In frontend programming, promises are often used for network requests. So let's see an extended example of that. -We'll use the [fetch](mdn:api/WindowOrWorkerGlobalScope/fetch) method to load the information about the user from the remote server. The method is quite complex, it has many optional parameters, but the basic usage is quite simple: +We'll use the [fetch](info:fetch) method to load the information about the user from the remote server. It has a lot of optional parameters covered in [separate chapters](info:fetch), but the basic syntax is quite simple: ```js let promise = fetch(url); @@ -252,7 +234,7 @@ let promise = fetch(url); This makes a network request to the `url` and returns a promise. The promise resolves with a `response` object when the remote server responds with headers, but *before the full response is downloaded*. -To read the full response, we should call a method `response.text()`: it returns a promise that resolves when the full text downloaded from the remote server, with that text as a result. +To read the full response, we should call the method `response.text()`: it returns a promise that resolves when the full text is downloaded from the remote server, with that text as a result. The code below makes a request to `user.json` and loads its text from the server: @@ -261,16 +243,16 @@ fetch('/article/promise-chaining/user.json') // .then below runs when the remote server responds .then(function(response) { // response.text() returns a new promise that resolves with the full response text - // when we finish downloading it + // when it loads return response.text(); }) .then(function(text) { // ...and here's the content of the remote file - alert(text); // {"name": "iliakan", isAdmin: true} + alert(text); // {"name": "iliakan", "isAdmin": true} }); ``` -There is also a method `response.json()` that reads the remote data and parses it as JSON. In our case that's even more convenient, so let's switch to it. +The `response` object returned from `fetch` also includes the method `response.json()` that reads the remote data and parses it as JSON. In our case that's even more convenient, so let's switch to it. We'll also use arrow functions for brevity: @@ -278,7 +260,7 @@ We'll also use arrow functions for brevity: // same as above, but response.json() parses the remote content as JSON fetch('/article/promise-chaining/user.json') .then(response => response.json()) - .then(user => alert(user.name)); // iliakan + .then(user => alert(user.name)); // iliakan, got user name ``` Now let's do something with the loaded user. @@ -305,7 +287,7 @@ fetch('/article/promise-chaining/user.json') }); ``` -The code works, see comments about the details, but it should be quite self-descriptive. Although, there's a potential problem in it, a typical error of those who begin to use promises. +The code works; see comments about the details. However, there's a potential problem in it, a typical error for those who begin to use promises. Look at the line `(*)`: how can we do something *after* the avatar has finished showing and gets removed? For instance, we'd like to show a form for editing that user or something else. As of now, there's no way. @@ -319,7 +301,7 @@ fetch('/article/promise-chaining/user.json') .then(user => fetch(`https://api.github.com/users/${user.name}`)) .then(response => response.json()) *!* - .then(githubUser => new Promise(function(resolve, reject) { + .then(githubUser => new Promise(function(resolve, reject) { // (*) */!* let img = document.createElement('img'); img.src = githubUser.avatar_url; @@ -329,7 +311,7 @@ fetch('/article/promise-chaining/user.json') setTimeout(() => { img.remove(); *!* - resolve(githubUser); + resolve(githubUser); // (**) */!* }, 3000); })) @@ -337,11 +319,9 @@ fetch('/article/promise-chaining/user.json') .then(githubUser => alert(`Finished showing ${githubUser.name}`)); ``` -Now right after `setTimeout` runs `img.remove()`, it calls `resolve(githubUser)`, thus passing the control to the next `.then` in the chain and passing forward the user data. +That is, the `.then` handler in line `(*)` now returns `new Promise`, that becomes settled only after the call of `resolve(githubUser)` in `setTimeout` `(**)`. The next `.then` in the chain will wait for that. -As a rule, an asynchronous action should always return a promise. - -That makes it possible to plan actions after it. Even if we don't plan to extend the chain now, we may need it later. +As a good practice, an asynchronous action should always return a promise. That makes it possible to plan actions after it; even if we don't plan to extend the chain now, we may need it later. Finally, we can split the code into reusable functions: @@ -352,8 +332,7 @@ function loadJson(url) { } function loadGithubUser(name) { - return fetch(`https://api.github.com/users/${name}`) - .then(response => response.json()); + return loadJson(`https://api.github.com/users/${name}`); } function showAvatar(githubUser) { diff --git a/1-js/11-async/03-promise-chaining/head.html b/1-js/11-async/03-promise-chaining/head.html index 31c6b4271..0a0075fb9 100644 --- a/1-js/11-async/03-promise-chaining/head.html +++ b/1-js/11-async/03-promise-chaining/head.html @@ -10,25 +10,6 @@ document.head.append(script); }); } - -class HttpError extends Error { - constructor(response) { - super(`${response.status} for ${response.url}`); - this.name = 'HttpError'; - this.response = response; - } -} - -function loadJson(url) { - return fetch(url) - .then(response => { - if (response.status == 200) { - return response.json(); - } else { - throw new HttpError(response); - } - }) -} </script> <style> diff --git a/1-js/11-async/04-promise-error-handling/article.md b/1-js/11-async/04-promise-error-handling/article.md index ca67202aa..c5b4206ab 100644 --- a/1-js/11-async/04-promise-error-handling/article.md +++ b/1-js/11-async/04-promise-error-handling/article.md @@ -1,11 +1,9 @@ # Error handling with promises -Asynchronous actions may sometimes fail: in case of an error the corresponding promise becomes rejected. For instance, `fetch` fails if the remote server is not available. We can use `.catch` to handle errors (rejections). +Promise chains are great at error handling. When a promise rejects, the control jumps to the closest rejection handler. That's very convenient in practice. -Promise chaining is great at that aspect. When a promise rejects, the control jumps to the closest rejection handler down the chain. That's very convenient in practice. - -For instance, in the code below the URL is wrong (no such server) and `.catch` handles the error: +For instance, in the code below the URL to `fetch` is wrong (no such site) and `.catch` handles the error: ```js run *!* @@ -15,17 +13,9 @@ fetch('https://no-such-server.blabla') // rejects .catch(err => alert(err)) // TypeError: failed to fetch (the text may vary) ``` -Or, maybe, everything is all right with the server, but the response is not valid JSON: +As you can see, the `.catch` doesn't have to be immediate. It may appear after one or maybe several `.then`. -```js run -fetch('/') // fetch works fine now, the server responds successfully -*!* - .then(response => response.json()) // rejects: the page is HTML, not a valid json -*/!* - .catch(err => alert(err)) // SyntaxError: Unexpected token < in JSON at position 0 -``` - -The easiest way to catch all errors is to append `.catch` to the end of chain: +Or, maybe, everything is all right with the site, but the response is not valid JSON. The easiest way to catch all errors is to append `.catch` to the end of chain: ```js run fetch('/article/promise-chaining/user.json') @@ -48,11 +38,11 @@ fetch('/article/promise-chaining/user.json') */!* ``` -Normally, `.catch` doesn't trigger at all, because there are no errors. But if any of the promises above rejects (a network problem or invalid json or whatever), then it would catch it. +Normally, such `.catch` doesn't trigger at all. But if any of the promises above rejects (a network problem or invalid json or whatever), then it would catch it. ## Implicit try..catch -The code of a promise executor and promise handlers has an "invisible `try..catch`" around it. If an error happens, it gets caught and treated as a rejection. +The code of a promise executor and promise handlers has an "invisible `try..catch`" around it. If an exception happens, it gets caught and treated as a rejection. For instance, this code: @@ -70,13 +60,13 @@ new Promise((resolve, reject) => { new Promise((resolve, reject) => { *!* reject(new Error("Whoops!")); -*/!* +*/!* }).catch(alert); // Error: Whoops! ``` -The "invisible `try..catch`" around the executor automatically catches the error and treats it as a rejection. +The "invisible `try..catch`" around the executor automatically catches the error and turns it into rejected promise. -This happens not only in the executor, but in its handlers as well. If we `throw` inside a `.then` handler, that means a rejected promise, so the control jumps to the nearest error handler. +This happens not only in the executor function, but in its handlers as well. If we `throw` inside a `.then` handler, that means a rejected promise, so the control jumps to the nearest error handler. Here's an example: @@ -102,15 +92,15 @@ new Promise((resolve, reject) => { }).catch(alert); // ReferenceError: blabla is not defined ``` -The final `.catch` not only catches explicit rejections, but also occasional errors in the handlers above. +The final `.catch` not only catches explicit rejections, but also accidental errors in the handlers above. ## Rethrowing -As we already noticed, `.catch` behaves like `try..catch`. We may have as many `.then` handlers as we want, and then use a single `.catch` at the end to handle errors in all of them. +As we already noticed, `.catch` at the end of the chain is similar to `try..catch`. We may have as many `.then` handlers as we want, and then use a single `.catch` at the end to handle errors in all of them. -In a regular `try..catch` we can analyze the error and maybe rethrow it if can't handle. The same thing is possible for promises. +In a regular `try..catch` we can analyze the error and maybe rethrow it if it can't be handled. The same thing is possible for promises. -If we `throw` inside `.catch`, then the control goes to the next closest error handler. And if we handle the error and finish normally, then it continues to the closest successful `.then` handler. +If we `throw` inside `.catch`, then the control goes to the next closest error handler. And if we handle the error and finish normally, then it continues to the next closest successful `.then` handler. In the example below the `.catch` successfully handles the error: @@ -120,7 +110,7 @@ new Promise((resolve, reject) => { throw new Error("Whoops!"); -}).catch(function(error) { +}).catch(function(error) { alert("The error is handled, continue normally"); @@ -132,7 +122,7 @@ Here the `.catch` block finishes normally. So the next successful `.then` handle In the example below we see the other situation with `.catch`. The handler `(*)` catches the error and just can't handle it (e.g. it only knows how to handle `URIError`), so it throws it again: ```js run -// the execution: catch -> catch -> then +// the execution: catch -> catch new Promise((resolve, reject) => { throw new Error("Whoops!"); @@ -150,7 +140,7 @@ new Promise((resolve, reject) => { } }).then(function() { - /* never runs here */ + /* doesn't run here */ }).catch(error => { // (**) alert(`The unknown error has occurred: ${error}`); @@ -159,116 +149,28 @@ new Promise((resolve, reject) => { }); ``` -Then the execution jumps from the first `.catch` `(*)` to the next one `(**)` down the chain. - -In the section below we'll see a practical example of rethrowing. - -## Fetch error handling example - -Let's improve error handling for the user-loading example. - -The promise returned by [fetch](mdn:api/WindowOrWorkerGlobalScope/fetch) rejects when it's impossible to make a request. For instance, a remote server is not available, or the URL is malformed. But if the remote server responds with error 404, or even error 500, then it's considered a valid response. - -What if the server returns a non-JSON page with error 500 in the line `(*)`? What if there's no such user, and github returns a page with error 404 at `(**)`? - -```js run -fetch('no-such-user.json') // (*) - .then(response => response.json()) - .then(user => fetch(`https://api.github.com/users/${user.name}`)) // (**) - .then(response => response.json()) - .catch(alert); // SyntaxError: Unexpected token < in JSON at position 0 - // ... -``` - - -As of now, the code tries to load the response as JSON no matter what and dies with a syntax error. You can see that by running the example above, as the file `no-such-user.json` doesn't exist. - -That's not good, because the error just falls through the chain, without details: what failed and where. - -So let's add one more step: we should check the `response.status` property that has HTTP status, and if it's not 200, then throw an error. - -```js run -class HttpError extends Error { // (1) - constructor(response) { - super(`${response.status} for ${response.url}`); - this.name = 'HttpError'; - this.response = response; - } -} - -function loadJson(url) { // (2) - return fetch(url) - .then(response => { - if (response.status == 200) { - return response.json(); - } else { - throw new HttpError(response); - } - }) -} - -loadJson('no-such-user.json') // (3) - .catch(alert); // HttpError: 404 for .../no-such-user.json -``` - -1. We make a custom class for HTTP Errors to distinguish them from other types of errors. Besides, the new class has a constructor that accepts `response` object and saves it in the error. So error-handling code will be able to access it. -2. Then we put together the requesting and error-handling code into a function that fetches the `url` *and* treats any non-200 status as an error. That's convenient, because we often need such logic. -3. Now `alert` shows a more helpful descriptive message. - -The great thing about having our own class for errors is that we can easily check for it in error-handling code. - -For instance, we can make a request, and then if we get 404 -- ask the user to modify the information. - -The code below loads a user with the given name from github. If there's no such user, then it asks for the correct name: - -```js run -function demoGithubUser() { - let name = prompt("Enter a name?", "iliakan"); - - return loadJson(`https://api.github.com/users/${name}`) - .then(user => { - alert(`Full name: ${user.name}.`); - return user; - }) - .catch(err => { -*!* - if (err instanceof HttpError && err.response.status == 404) { -*/!* - alert("No such user, please reenter."); - return demoGithubUser(); - } else { - throw err; // (*) - } - }); -} - -demoGithubUser(); -``` - -Please note: `.catch` here catches all errors, but it "knows how to handle" only `HttpError 404`. In that particular case it means that there's no such user, and `.catch` just retries in that case. - -For other errors, it has no idea what could go wrong. Maybe a programming error or something. So it just rethrows it in the line `(*)`. +The execution jumps from the first `.catch` `(*)` to the next one `(**)` down the chain. ## Unhandled rejections -What happens when an error is not handled? For instance, after the rethrow `(*)` in the example above. - -Or we could just forget to append an error handler to the end of the chain, like here: +What happens when an error is not handled? For instance, we forgot to append `.catch` to the end of the chain, like here: ```js untrusted run refresh new Promise(function() { noSuchFunction(); // Error here (no such function) }) .then(() => { - // zero or many promise handlers + // successful promise handlers, one or more }); // without .catch at the end! ``` -In case of an error, the promise state becomes "rejected", and the execution should jump to the closest rejection handler. But there is no such handler in the examples above. So the error gets "stuck". +In case of an error, the promise becomes rejected, and the execution should jump to the closest rejection handler. But there is none. So the error gets "stuck". There's no code to handle it. + +In practice, just like with regular unhandled errors in code, it means that something has gone terribly wrong. -In practice, just like with a regular unhandled errors, it means that something has terribly gone wrong, the script probably died. +What happens when a regular error occurs and is not caught by `try..catch`? The script dies with a message in the console. A similar thing happens with unhandled promise rejections. -Most JavaScript engines track such situations and generate a global error in that case. We can see it in the console. +The JavaScript engine tracks such rejections and generates a global error in that case. You can see it in the console if you run the example above. In the browser we can catch such errors using the event `unhandledrejection`: @@ -292,52 +194,12 @@ If an error occurs, and there's no `.catch`, the `unhandledrejection` handler tr Usually such errors are unrecoverable, so our best way out is to inform the user about the problem and probably report the incident to the server. -In non-browser environments like Node.js there are other similar ways to track unhandled errors. - +In non-browser environments like Node.js there are other ways to track unhandled errors. ## Summary -- `.catch` handles promise rejections of all kinds: be it a `reject()` call, or an error thrown in a handler. -- We should place `.catch` exactly in places where we want to handle errors and know how to handle them. The handler should analyze errors (custom error classes help) and rethrow unknown ones. -- It's normal not to use `.catch` if we don't know how to handle errors (all errors are unrecoverable). -- In any case we should have the `unhandledrejection` event handler (for browsers, and analogs for other environments), to track unhandled errors and inform the user (and probably our server) about the them, so that our app never "just dies". - -And finally, if we have load-indication, then `.finally` is a great handler to stop it when the fetch is complete: - -```js run -function demoGithubUser() { - let name = prompt("Enter a name?", "iliakan"); - -*!* - document.body.style.opacity = 0.3; // (1) start the indication -*/!* - - return loadJson(`https://api.github.com/users/${name}`) -*!* - .finally(() => { // (2) stop the indication - document.body.style.opacity = ''; - return new Promise(resolve => setTimeout(resolve, 0)); // (*) - }) -*/!* - .then(user => { - alert(`Full name: ${user.name}.`); - return user; - }) - .catch(err => { - if (err instanceof HttpError && err.response.status == 404) { - alert("No such user, please reenter."); - return demoGithubUser(); - } else { - throw err; - } - }); -} - -demoGithubUser(); -``` - -Here on the line `(1)` we indicate loading by dimming the document. The method doesn't matter, could use any type of indication instead. - -When the promise is settled, be it a successful fetch or an error, `finally` triggers at the line `(2)` and stops the indication. - -There's a little browser trick `(*)` with returning a zero-timeout promise from `finally`. That's because some browsers (like Chrome) need "a bit time" outside promise handlers to paint document changes. So it ensures that the indication is visually stopped before going further on the chain. +- `.catch` handles errors in promises of all kinds: be it a `reject()` call, or an error thrown in a handler. +- `.then` also catches errors in the same manner, if given the second argument (which is the error handler). +- We should place `.catch` exactly in places where we want to handle errors and know how to handle them. The handler should analyze errors (custom error classes help) and rethrow unknown ones (maybe they are programming mistakes). +- It's ok not to use `.catch` at all, if there's no way to recover from an error. +- In any case we should have the `unhandledrejection` event handler (for browsers, and analogs for other environments) to track unhandled errors and inform the user (and probably our server) about them, so that our app never "just dies". diff --git a/1-js/11-async/04-promise-error-handling/head.html b/1-js/11-async/04-promise-error-handling/head.html index 31c6b4271..a0b741962 100644 --- a/1-js/11-async/04-promise-error-handling/head.html +++ b/1-js/11-async/04-promise-error-handling/head.html @@ -1,16 +1,4 @@ <script> -function loadScript(src) { - return new Promise(function(resolve, reject) { - let script = document.createElement('script'); - script.src = src; - - script.onload = () => resolve(script); - script.onerror = () => reject(new Error("Script load error: " + src)); - - document.head.append(script); - }); -} - class HttpError extends Error { constructor(response) { super(`${response.status} for ${response.url}`); diff --git a/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.md b/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.md deleted file mode 100644 index 17ac6ba5d..000000000 --- a/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.md +++ /dev/null @@ -1,44 +0,0 @@ -The solution is actually pretty simple. - -Take a look at this: - -```js -Promise.all( - fetch('https://api.github.com/users/iliakan'), - fetch('https://api.github.com/users/remy'), - fetch('http://no-such-url') -) -``` - -Here we have an array of `fetch(...)` promises that goes to `Promise.all`. - -We can't change the way `Promise.all` works: if it detects an error, then it rejects with it. So we need to prevent any error from occurring. Instead, if a `fetch` error happens, we need to treat it as a "normal" result. - -Here's how: - -```js -Promise.all( - fetch('https://api.github.com/users/iliakan').catch(err => err), - fetch('https://api.github.com/users/remy').catch(err => err), - fetch('http://no-such-url').catch(err => err) -) -``` - -In other words, the `.catch` takes an error for all of the promises and returns it normally. By the rules of how promises work, if a `.then/catch` handler returns a value (doesn't matter if it's an error object or something else), then the execution continues the "normal" flow. - -So the `.catch` returns the error as a "normal" result into the outer `Promise.all`. - -This code: -```js -Promise.all( - urls.map(url => fetch(url)) -) -``` - -Can be rewritten as: - -```js -Promise.all( - urls.map(url => fetch(url).catch(err => err)) -) -``` diff --git a/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.view/index.html b/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.view/index.html deleted file mode 100644 index 209e60640..000000000 --- a/1-js/11-async/05-promise-api/01-promise-errors-as-results/solution.view/index.html +++ /dev/null @@ -1,24 +0,0 @@ -<!DOCTYPE HTML> -<html> -<body> - - <script> - - let urls = [ - 'https://api.github.com/users/iliakan', - 'https://api.github.com/users/remy', - 'http://no-such-url' - ]; - - Promise.all( - urls.map(url => fetch(url).catch(err => err)) - ) - .then(responses => { - alert(responses[0].status); // 200 - alert(responses[1].status); // 200 - alert(responses[2]); // TypeError: failed to fetch (text may vary) - }); - </script> - -</body> -</html> diff --git a/1-js/11-async/05-promise-api/01-promise-errors-as-results/source.view/index.html b/1-js/11-async/05-promise-api/01-promise-errors-as-results/source.view/index.html deleted file mode 100644 index c760ad8cd..000000000 --- a/1-js/11-async/05-promise-api/01-promise-errors-as-results/source.view/index.html +++ /dev/null @@ -1,28 +0,0 @@ -<!DOCTYPE HTML> -<html> -<body> - - <script> - - // if any of URLs fails, other results are ignored - // change that: - // make errors appear as members of the responses array, together with normal results - - let urls = [ - 'https://api.github.com/users/iliakan', - 'https://api.github.com/users/remy', - 'http://no-such-url' - ]; - - // Fix me: - Promise.all(urls.map(url => fetch(url))) - // Demo output (no need to change): - .then(responses => { - alert(responses[0].status); // 200 - alert(responses[1].status); // 200 - alert(responses[2]); // TypeError: failed to fetch (text may vary) - }); - </script> - -</body> -</html> diff --git a/1-js/11-async/05-promise-api/01-promise-errors-as-results/task.md b/1-js/11-async/05-promise-api/01-promise-errors-as-results/task.md deleted file mode 100644 index c37ad56e3..000000000 --- a/1-js/11-async/05-promise-api/01-promise-errors-as-results/task.md +++ /dev/null @@ -1,48 +0,0 @@ -# Fault-tolerant Promise.all - -We'd like to fetch multiple URLs in parallel. - -Here's the code to do that: - -```js run -let urls = [ - 'https://api.github.com/users/iliakan', - 'https://api.github.com/users/remy', - 'https://api.github.com/users/jeresig' -]; - -Promise.all(urls.map(url => fetch(url))) - // for each response show its status - .then(responses => { // (*) - for(let response of responses) { - alert(`${response.url}: ${response.status}`); - } - }); -``` - -The problem is that if any of requests fails, then `Promise.all` rejects with the error, and we lose the results of all the other requests. - -That's not good. - -Modify the code so that the array `responses` in the line `(*)` would include the response objects for successful fetches and error objects for failed ones. - -For instance, if one of the URLs is bad, then it should be like: - -```js -let urls = [ - 'https://api.github.com/users/iliakan', - 'https://api.github.com/users/remy', - 'http://no-such-url' -]; - -Promise.all(...) // your code to fetch URLs... - // ...and pass fetch errors as members of the resulting array... - .then(responses => { - // 3 urls => 3 array members - alert(responses[0].status); // 200 - alert(responses[1].status); // 200 - alert(responses[2]); // TypeError: failed to fetch (text may vary) - }); -``` - -P.S. In this task you don't have to load the full response using `response.text()` or `response.json()`. Just handle fetch errors the right way. diff --git a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/solution.view/index.html b/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/solution.view/index.html deleted file mode 100644 index 744efd2b7..000000000 --- a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/solution.view/index.html +++ /dev/null @@ -1,29 +0,0 @@ -<!DOCTYPE HTML> -<html> -<body> - - <script> - - let urls = [ - 'https://api.github.com/users/iliakan', - '/', - 'http://no-such-url' - ]; - - Promise.all( - urls.map(url => fetch(url).catch(err => err)) - ) - .then(responses => Promise.all( - // if it's an error then pass on - // otherwise response.json() and catch errors as results - responses.map(r => r instanceof Error ? r : r.json().catch(err => err)) - )) - .then(results => { - alert(results[0].name); // Ilya Kantor - alert(results[1]); // SyntaxError: Unexpected token < in JSON at position 0 - alert(results[2]); // TypeError: failed to fetch (text may vary) - }); - </script> - -</body> -</html> diff --git a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/source.view/index.html b/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/source.view/index.html deleted file mode 100644 index adb86d41f..000000000 --- a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/source.view/index.html +++ /dev/null @@ -1,33 +0,0 @@ -<!DOCTYPE HTML> -<html> -<body> - - <script> - - // the whole promise chain fails with an error here - // change that: - // make errors appear as members of the results array - - let urls = [ - 'https://api.github.com/users/iliakan', - // this URL is HTML page, it's invalid JSON, so response.json() fails - '/', - // this URL is invalid, so fetch fails - 'http://no-such-url' - ]; - - // Fix it: - Promise.all(urls.map(url => fetch(url))) - .then(responses => Promise.all( - responses.map(r => r.json()) - )) - // Demo output (no need to change): - .then(results => { - alert(results[0].name); // Ilya Kantor - alert(results[1]); // SyntaxError: Unexpected token < in JSON at position 0 - alert(results[2]); // TypeError: failed to fetch (text may vary) - }); - </script> - -</body> -</html> diff --git a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/task.md b/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/task.md deleted file mode 100644 index d28cf87e7..000000000 --- a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/task.md +++ /dev/null @@ -1,34 +0,0 @@ -# Fault-tolerant fetch with JSON - -Improve the solution of the previous task <info:task/promise-errors-as-results>. Now we need not just to call `fetch`, but to load the JSON objects from the given URLs. - -Here's the example code to do that: - -```js run -let urls = [ - 'https://api.github.com/users/iliakan', - 'https://api.github.com/users/remy', - 'https://api.github.com/users/jeresig' -]; - -// make fetch requests -Promise.all(urls.map(url => fetch(url))) - // map each response to response.json() - .then(responses => Promise.all( - responses.map(r => r.json()) - )) - // show name of each user - .then(users => { // (*) - for(let user of users) { - alert(user.name); - } - }); -``` - -The problem is that if any of requests fails, then `Promise.all` rejects with the error, and we lose results of all the other requests. So the code above is not fault-tolerant, just like the one in the previous task. - -Modify the code so that the array in the line `(*)` would include parsed JSON for successful requests and error for errored ones. - -Please note that the error may occur both in `fetch` (if the network request fails) and in `response.json()` (if the response is invalid JSON). In both cases the error should become a member of the results object. - -The sandbox has both of these cases. diff --git a/1-js/11-async/05-promise-api/article.md b/1-js/11-async/05-promise-api/article.md index d049f3b09..7be84ce2c 100644 --- a/1-js/11-async/05-promise-api/article.md +++ b/1-js/11-async/05-promise-api/article.md @@ -1,83 +1,24 @@ # Promise API -There are 4 static methods in the `Promise` class. We'll quickly cover their use cases here. - -## Promise.resolve - -The syntax: - -```js -let promise = Promise.resolve(value); -``` - -Returns a resolved promise with the given `value`. - -Same as: - -```js -let promise = new Promise(resolve => resolve(value)); -``` - -The method is used when we already have a value, but would like to have it "wrapped" into a promise. - -For instance, the `loadCached` function below fetches the `url` and remembers the result, so that future calls on the same URL return it immediately: - -```js -function loadCached(url) { - let cache = loadCached.cache || (loadCached.cache = new Map()); - - if (cache.has(url)) { -*!* - return Promise.resolve(cache.get(url)); // (*) -*/!* - } - - return fetch(url) - .then(response => response.text()) - .then(text => { - cache.set(url,text); - return text; - }); -} -``` - -We can use `loadCached(url).then(…)`, because the function is guaranteed to return a promise. That's the purpose `Promise.resolve` serves in the line `(*)`: it makes sure the interface is unified. We can always use `.then` after `loadCached`. - -## Promise.reject - -The syntax: - -```js -let promise = Promise.reject(error); -``` - -Create a rejected promise with the `error`. - -Same as: - -```js -let promise = new Promise((resolve, reject) => reject(error)); -``` - -We cover it here for completeness, rarely used in real code. +There are 6 static methods in the `Promise` class. We'll quickly cover their use cases here. ## Promise.all -Let's say we want to run many promises to execute in parallel, and wait till all of them are ready. +Let's say we want many promises to execute in parallel and wait until all of them are ready. -For instance, download several URLs in parallel and process the content when all are done. +For instance, download several URLs in parallel and process the content once they are all done. That's what `Promise.all` is for. The syntax is: ```js -let promise = Promise.all([...promises...]); +let promise = Promise.all(iterable); ``` -It takes an array of promises (technically can be any iterable, but usually an array) and returns a new promise. +`Promise.all` takes an iterable (usually, an array of promises) and returns a new promise. -The new promise resolves when all listed promises are settled and has an array of their results. +The new promise resolves when all listed promises are resolved, and the array of their results becomes its result. For instance, the `Promise.all` below settles after 3 seconds, and then its result is an array `[1, 2, 3]`: @@ -89,7 +30,7 @@ Promise.all([ ]).then(alert); // 1,2,3 when promises are ready: each promise contributes an array member ``` -Please note that the relative order is the same. Even though the first promise takes the longest time to resolve, it is still first in the array of results. +Please note that the order of the resulting array members is the same as in its source promises. Even though the first promise takes the longest time to resolve, it's still first in the array of results. A common trick is to map an array of job data into an array of promises, and then wrap that into `Promise.all`. @@ -102,7 +43,7 @@ let urls = [ 'https://api.github.com/users/jeresig' ]; -// map every url to the promise fetch(github url) +// map every url to the promise of the fetch let requests = urls.map(url => fetch(url)); // Promise.all waits until all jobs are resolved @@ -112,7 +53,7 @@ Promise.all(requests) )); ``` -A more real-life example with fetching user information for an array of github users by their names (or we could fetch an array of goods by their ids, the logic is same): +A bigger example with fetching user information for an array of GitHub users by their names (we could fetch an array of goods by their ids, the logic is identical): ```js run let names = ['iliakan', 'remy', 'jeresig']; @@ -121,20 +62,20 @@ let requests = names.map(name => fetch(`https://api.github.com/users/${name}`)); Promise.all(requests) .then(responses => { - // all responses are ready, we can show HTTP status codes + // all responses are resolved successfully for(let response of responses) { alert(`${response.url}: ${response.status}`); // shows 200 for every url } return responses; }) - // map array of responses into array of response.json() to read their content + // map array of responses into an array of response.json() to read their content .then(responses => Promise.all(responses.map(r => r.json()))) // all JSON answers are parsed: "users" is the array of them .then(users => users.forEach(user => alert(user.name))); ``` -If any of the promises is rejected, `Promise.all` immediately rejects with that error. +**If any of the promises is rejected, the promise returned by `Promise.all` immediately rejects with that error.** For instance: @@ -148,14 +89,18 @@ Promise.all([ ]).catch(alert); // Error: Whoops! ``` -Here the second promise rejects in two seconds. That leads to immediate rejection of `Promise.all`, so `.catch` executes: the rejection error becomes the outcome of the whole `Promise.all`. +Here the second promise rejects in two seconds. That leads to an immediate rejection of `Promise.all`, so `.catch` executes: the rejection error becomes the outcome of the entire `Promise.all`. -The important detail is that promises provide no way to "cancel" or "abort" their execution. So other promises continue to execute, and then eventually settle, but all their results are ignored. +```warn header="In case of an error, other promises are ignored" +If one promise rejects, `Promise.all` immediately rejects, completely forgetting about the other ones in the list. Their results are ignored. -There are ways to avoid this: we can either write additional code to `clearTimeout` (or otherwise cancel) the promises in case of an error, or we can make errors show up as members in the resulting array (see the task below this chapter about it). +For example, if there are multiple `fetch` calls, like in the example above, and one fails, the others will still continue to execute, but `Promise.all` won't watch them anymore. They will probably settle, but their results will be ignored. -````smart header="`Promise.all(...)` allows non-promise items in `iterable`" -Normally, `Promise.all(...)` accepts an iterable (in most cases an array) of promises. But if any of those objects is not a promise, it's wrapped in `Promise.resolve`. +`Promise.all` does nothing to cancel them, as there's no concept of "cancellation" in promises. In [another chapter](info:fetch-abort) we'll cover `AbortController` that can help with that, but it's not a part of the Promise API. +``` + +````smart header="`Promise.all(iterable)` allows non-promise \"regular\" values in `iterable`" +Normally, `Promise.all(...)` accepts an iterable (in most cases an array) of promises. But if any of those objects is not a promise, it's passed to the resulting array "as is". For instance, here the results are `[1, 2, 3]`: @@ -164,18 +109,94 @@ Promise.all([ new Promise((resolve, reject) => { setTimeout(() => resolve(1), 1000) }), - 2, // treated as Promise.resolve(2) - 3 // treated as Promise.resolve(3) + 2, + 3 ]).then(alert); // 1, 2, 3 ``` -So we are able to pass non-promise values to `Promise.all` where convenient. - +So we are able to pass ready values to `Promise.all` where convenient. ```` +## Promise.allSettled + +[recent browser="new"] + +`Promise.all` rejects as a whole if any promise rejects. That's good for "all or nothing" cases, when we need *all* results successful to proceed: + +```js +Promise.all([ + fetch('/template.html'), + fetch('/style.css'), + fetch('/data.json') +]).then(render); // render method needs results of all fetches +``` + +`Promise.allSettled` just waits for all promises to settle, regardless of the result. The resulting array has: + +- `{status:"fulfilled", value:result}` for successful responses, +- `{status:"rejected", reason:error}` for errors. + +For example, we'd like to fetch the information about multiple users. Even if one request fails, we're still interested in the others. + +Let's use `Promise.allSettled`: + +```js run +let urls = [ + 'https://api.github.com/users/iliakan', + 'https://api.github.com/users/remy', + 'https://no-such-url' +]; + +Promise.allSettled(urls.map(url => fetch(url))) + .then(results => { // (*) + results.forEach((result, num) => { + if (result.status == "fulfilled") { + alert(`${urls[num]}: ${result.value.status}`); + } + if (result.status == "rejected") { + alert(`${urls[num]}: ${result.reason}`); + } + }); + }); +``` + +The `results` in the line `(*)` above will be: +```js +[ + {status: 'fulfilled', value: ...response...}, + {status: 'fulfilled', value: ...response...}, + {status: 'rejected', reason: ...error object...} +] +``` + +So for each promise we get its status and `value/error`. + +### Polyfill + +If the browser doesn't support `Promise.allSettled`, it's easy to polyfill: + +```js +if (!Promise.allSettled) { + const rejectHandler = reason => ({ status: 'rejected', reason }); + + const resolveHandler = value => ({ status: 'fulfilled', value }); + + Promise.allSettled = function (promises) { + const convertedPromises = promises.map(p => Promise.resolve(p).then(resolveHandler, rejectHandler)); + return Promise.all(convertedPromises); + }; +} +``` + +In this code, `promises.map` takes input values, turns them into promises (just in case a non-promise was passed) with `p => Promise.resolve(p)`, and then adds `.then` handler to every one. + +That handler turns a successful result `value` into `{status:'fulfilled', value}`, and an error `reason` into `{status:'rejected', reason}`. That's exactly the format of `Promise.allSettled`. + +Now we can use `Promise.allSettled` to get the results of *all* given promises, even if some of them reject. + ## Promise.race -Similar to `Promise.all`, it takes an iterable of promises, but instead of waiting for all of them to finish, it waits for the first result (or error), and goes on with it. +Similar to `Promise.all`, but waits only for the first settled promise and gets its result (or error). The syntax is: @@ -193,15 +214,110 @@ Promise.race([ ]).then(alert); // 1 ``` -So, the first result/error becomes the result of the whole `Promise.race`. After the first settled promise "wins the race", all further results/errors are ignored. +The first promise here was fastest, so it became the result. After the first settled promise "wins the race", all further results/errors are ignored. + + +## Promise.any + +Similar to `Promise.race`, but waits only for the first fulfilled promise and gets its result. If all of the given promises are rejected, then the returned promise is rejected with [`AggregateError`](mdn:js/AggregateError) - a special error object that stores all promise errors in its `errors` property. + +The syntax is: + +```js +let promise = Promise.any(iterable); +``` + +For instance, here the result will be `1`: + +```js run +Promise.any([ + new Promise((resolve, reject) => setTimeout(() => reject(new Error("Whoops!")), 1000)), + new Promise((resolve, reject) => setTimeout(() => resolve(1), 2000)), + new Promise((resolve, reject) => setTimeout(() => resolve(3), 3000)) +]).then(alert); // 1 +``` + +The first promise here was fastest, but it was rejected, so the second promise became the result. After the first fulfilled promise "wins the race", all further results are ignored. + +Here's an example when all promises fail: + +```js run +Promise.any([ + new Promise((resolve, reject) => setTimeout(() => reject(new Error("Ouch!")), 1000)), + new Promise((resolve, reject) => setTimeout(() => reject(new Error("Error!")), 2000)) +]).catch(error => { + console.log(error.constructor.name); // AggregateError + console.log(error.errors[0]); // Error: Ouch! + console.log(error.errors[1]); // Error: Error! +}); +``` + +As you can see, error objects for failed promises are available in the `errors` property of the `AggregateError` object. + +## Promise.resolve/reject + +Methods `Promise.resolve` and `Promise.reject` are rarely needed in modern code, because `async/await` syntax (we'll cover it [a bit later](info:async-await)) makes them somewhat obsolete. + +We cover them here for completeness and for those who can't use `async/await` for some reason. + +### Promise.resolve + +`Promise.resolve(value)` creates a resolved promise with the result `value`. + +Same as: + +```js +let promise = new Promise(resolve => resolve(value)); +``` + +The method is used for compatibility, when a function is expected to return a promise. + +For example, the `loadCached` function below fetches a URL and remembers (caches) its content. For future calls with the same URL it immediately gets the previous content from cache, but uses `Promise.resolve` to make a promise of it, so the returned value is always a promise: + +```js +let cache = new Map(); + +function loadCached(url) { + if (cache.has(url)) { +*!* + return Promise.resolve(cache.get(url)); // (*) +*/!* + } + + return fetch(url) + .then(response => response.text()) + .then(text => { + cache.set(url,text); + return text; + }); +} +``` + +We can write `loadCached(url).then(…)`, because the function is guaranteed to return a promise. We can always use `.then` after `loadCached`. That's the purpose of `Promise.resolve` in the line `(*)`. + +### Promise.reject + +`Promise.reject(error)` creates a rejected promise with `error`. + +Same as: + +```js +let promise = new Promise((resolve, reject) => reject(error)); +``` + +In practice, this method is almost never used. ## Summary -There are 4 static methods of `Promise` class: +There are 6 static methods of `Promise` class: -1. `Promise.resolve(value)` -- makes a resolved promise with the given value. -2. `Promise.reject(error)` -- makes a rejected promise with the given error. -3. `Promise.all(promises)` -- waits for all promises to resolve and returns an array of their results. If any of the given promises rejects, then it becomes the error of `Promise.all`, and all other results are ignored. -4. `Promise.race(promises)` -- waits for the first promise to settle, and its result/error becomes the outcome. +1. `Promise.all(promises)` -- waits for all promises to resolve and returns an array of their results. If any of the given promises rejects, it becomes the error of `Promise.all`, and all other results are ignored. +2. `Promise.allSettled(promises)` (recently added method) -- waits for all promises to settle and returns their results as an array of objects with: + - `status`: `"fulfilled"` or `"rejected"` + - `value` (if fulfilled) or `reason` (if rejected). +3. `Promise.race(promises)` -- waits for the first promise to settle, and its result/error becomes the outcome. +4. `Promise.any(promises)` (recently added method) -- waits for the first promise to fulfill, and its result becomes the outcome. If all of the given promises are rejected, [`AggregateError`](mdn:js/AggregateError) becomes the error of `Promise.any`. +5. `Promise.resolve(value)` -- makes a resolved promise with the given value. +6. `Promise.reject(error)` -- makes a rejected promise with the given error. -Of these four, `Promise.all` is the most common in practice. +Of all these, `Promise.all` is probably the most common in practice. diff --git a/1-js/11-async/06-promisify/article.md b/1-js/11-async/06-promisify/article.md index e72329ce1..855678e5b 100644 --- a/1-js/11-async/06-promisify/article.md +++ b/1-js/11-async/06-promisify/article.md @@ -1,10 +1,10 @@ # Promisification -Promisification -- is a long word for a simple transform. It's conversion of a function that accepts a callback into a function returning a promise. +"Promisification" is a long word for a simple transformation. It's the conversion of a function that accepts a callback into a function that returns a promise. -In other words, we create a wrapper-function that does the same, internally calling the original one, but returns a promise. +Such transformations are often required in real-life, as many functions and libraries are callback-based. But promises are more convenient, so it makes sense to promisify them. -Such transforms are often needed in real-life, as many functions and libraries are callback-based. But promises are more convenient. So it makes sense to promisify those. +For better understanding, let's see an example. For instance, we have `loadScript(src, callback)` from the chapter <info:callbacks>. @@ -23,61 +23,73 @@ function loadScript(src, callback) { // loadScript('path/script.js', (err, script) => {...}) ``` -Let's promisify it. The new `loadScriptPromise(src)` function will do the same, but accept only `src` (no callback) and return a promise. +The function loads a script with the given `src`, and then calls `callback(err)` in case of an error, or `callback(null, script)` in case of successful loading. That's a widespread agreement for using callbacks, we saw it before. +Let's promisify it. + +We'll make a new function `loadScriptPromise(src)`, that does the same (loads the script), but returns a promise instead of using callbacks. + +In other words, we pass it only `src` (no `callback`) and get a promise in return, that resolves with `script` when the load is successful, and rejects with the error otherwise. + +Here it is: ```js let loadScriptPromise = function(src) { return new Promise((resolve, reject) => { loadScript(src, (err, script) => { - if (err) reject(err) + if (err) reject(err); else resolve(script); }); - }) -} + }); +}; // usage: // loadScriptPromise('path/script.js').then(...) ``` -Now `loadScriptPromise` fits well in our promise-based code. - -As we can see, it delegates all the work to the original `loadScript`, providing its own callback that translates to promise `resolve/reject`. +As we can see, the new function is a wrapper around the original `loadScript` function. It calls it providing its own callback that translates to promise `resolve/reject`. -As we may need to promisify many functions, it makes sense to use a helper. +Now `loadScriptPromise` fits well in promise-based code. If we like promises more than callbacks (and soon we'll see more reasons for that), then we will use it instead. -That's actually very simple -- `promisify(f)` below takes a to-promisify function `f` and returns a wrapper function. +In practice we may need to promisify more than one function, so it makes sense to use a helper. -That wrapper does the same as in the code above: returns a promise and passes the call to the original `f`, tracking the result in a custom callback: +We'll call it `promisify(f)`: it accepts a to-promisify function `f` and returns a wrapper function. ```js function promisify(f) { - return function (...args) { // return a wrapper-function + return function (...args) { // return a wrapper-function (*) return new Promise((resolve, reject) => { - function callback(err, result) { // our custom callback for f + function callback(err, result) { // our custom callback for f (**) if (err) { - return reject(err); + reject(err); } else { resolve(result); } } - args.push(callback); // append our custom callback to the end of arguments + args.push(callback); // append our custom callback to the end of f arguments f.call(this, ...args); // call the original function }); }; -}; +} // usage: let loadScriptPromise = promisify(loadScript); loadScriptPromise(...).then(...); ``` -Here we assume that the original function expects a callback with two arguments `(err, result)`. That's what we encounter most often. Then our custom callback is in exactly the right format, and `promisify` works great for such a case. +The code may look a bit complex, but it's essentially the same that we wrote above, while promisifying `loadScript` function. + +A call to `promisify(f)` returns a wrapper around `f` `(*)`. That wrapper returns a promise and forwards the call to the original `f`, tracking the result in the custom callback `(**)`. + +Here, `promisify` assumes that the original function expects a callback with exactly two arguments `(err, result)`. That's what we encounter most often. Then our custom callback is in exactly the right format, and `promisify` works great for such a case. + +But what if the original `f` expects a callback with more arguments `callback(err, res1, res2, ...)`? -But what if the original `f` expects a callback with more arguments `callback(err, res1, res2)`? +We can improve our helper. Let's make a more advanced version of `promisify`. -Here's a modification of `promisify` that returns an array of multiple callback results: +- When called as `promisify(f)` it should work similar to the version above. +- When called as `promisify(f, true)`, it should return the promise that resolves with the array of callback results. That's exactly for callbacks with many arguments. ```js // promisify(f, true) to get array of results @@ -86,7 +98,7 @@ function promisify(f, manyArgs = false) { return new Promise((resolve, reject) => { function *!*callback(err, ...results*/!*) { // our custom callback for f if (err) { - return reject(err); + reject(err); } else { // resolve with all callback results if manyArgs is specified *!*resolve(manyArgs ? results : results[0]);*/!* @@ -98,19 +110,21 @@ function promisify(f, manyArgs = false) { f.call(this, ...args); }); }; -}; +} // usage: f = promisify(f, true); -f(...).then(arrayOfResults => ..., err => ...) +f(...).then(arrayOfResults => ..., err => ...); ``` -In some cases, `err` may be absent at all: `callback(result)`, or there's something exotic in the callback format, then we can promisify such functions manually. +As you can see it's essentially the same as above, but `resolve` is called with only one or all arguments depending on whether `manyArgs` is truthy. + +For more exotic callback formats, like those without `err` at all: `callback(result)`, we can promisify such functions manually without using the helper. There are also modules with a bit more flexible promisification functions, e.g. [es6-promisify](https://github.com/digitaldesignlabs/es6-promisify). In Node.js, there's a built-in `util.promisify` function for that. ```smart -Promisification is a great approach, especially when you use `async/await` (see the next chapter), but not a total replacement for callbacks. +Promisification is a great approach, especially when you use `async/await` (covered later in the chapter <info:async-await>), but not a total replacement for callbacks. Remember, a promise may have only one result, but a callback may technically be called many times. diff --git a/1-js/11-async/07-microtask-queue/article.md b/1-js/11-async/07-microtask-queue/article.md index 296fc977b..6a67c5e90 100644 --- a/1-js/11-async/07-microtask-queue/article.md +++ b/1-js/11-async/07-microtask-queue/article.md @@ -1,46 +1,46 @@ -# Microtasks and event loop +# Microtasks Promise handlers `.then`/`.catch`/`.finally` are always asynchronous. -Even when a Promise is immediately resolved, the code on the lines *below* your `.then`/`.catch`/`.finally` will still execute first. +Even when a Promise is immediately resolved, the code on the lines *below* `.then`/`.catch`/`.finally` will still execute before these handlers. -Here's the code that demonstrates it: +Here's a demo: ```js run let promise = Promise.resolve(); -promise.then(() => alert("promise done")); +promise.then(() => alert("promise done!")); alert("code finished"); // this alert shows first ``` -If you run it, you see `code finished` first, and then `promise done`. +If you run it, you see `code finished` first, and then `promise done!`. That's strange, because the promise is definitely done from the beginning. Why did the `.then` trigger afterwards? What's going on? -# Microtasks +## Microtasks queue -Asynchronous tasks need proper management. For that, the standard specifies an internal queue `PromiseJobs`, more often referred to as "microtask queue" (v8 term). +Asynchronous tasks need proper management. For that, the ECMA standard specifies an internal queue `PromiseJobs`, more often referred to as the "microtask queue" (V8 term). -As said in the [specification](https://tc39.github.io/ecma262/#sec-jobs-and-job-queues): +As stated in the [specification](https://tc39.github.io/ecma262/#sec-jobs-and-job-queues): - The queue is first-in-first-out: tasks enqueued first are run first. - Execution of a task is initiated only when nothing else is running. -Or, to say that simply, when a promise is ready, its `.then/catch/finally` handlers are put into the queue. They are not executed yet. JavaScript engine takes a task from the queue and executes it, when it becomes free from the current code. +Or, to put it more simply, when a promise is ready, its `.then/catch/finally` handlers are put into the queue; they are not executed yet. When the JavaScript engine becomes free from the current code, it takes a task from the queue and executes it. That's why "code finished" in the example above shows first.  -Promise handlers always go through that internal queue. +Promise handlers always go through this internal queue. -If there's a chain with multiple `.then/catch/finally`, then every one of them is executed asynchronously. That is, it first gets queued, and executed when the current code is complete and previously queued handlers are finished. +If there's a chain with multiple `.then/catch/finally`, then every one of them is executed asynchronously. That is, it first gets queued, then executed when the current code is complete and previously queued handlers are finished. -**What if the order matters for us? How can we make `code finished` work after `promise done`?** +**What if the order matters for us? How can we make `code finished` appear after `promise done`?** Easy, just put it into the queue with `.then`: @@ -52,6 +52,7 @@ Promise.resolve() Now the order is as intended. +<<<<<<< HEAD ## Event loop In-browser JavaScript, as well as Node.js, is based on an *event loop*. @@ -123,68 +124,63 @@ Naturally, `promise` shows up first, because `setTimeout` macrotask awaits in th As a logical consequence, macrotasks are handled only when promises give the engine a "free time". So if we have a promise chain that doesn't wait for anything, then things like `setTimeout` or event handlers can never get in the middle. +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Unhandled rejection -Remember "unhandled rejection" event from the chapter <info:promise-error-handling>? +Remember the `unhandledrejection` event from the article <info:promise-error-handling>? -Now, with the understanding of microtasks, we can formalize it. +Now we can see exactly how JavaScript finds out that there was an unhandled rejection. -**"Unhandled rejection" is when a promise error is not handled at the end of the microtask queue.** +**An "unhandled rejection" occurs when a promise error is not handled at the end of the microtask queue.** -For instance, consider this code: +Normally, if we expect an error, we add `.catch` to the promise chain to handle it: ```js run let promise = Promise.reject(new Error("Promise Failed!")); +*!* +promise.catch(err => alert('caught')); +*/!* -window.addEventListener('unhandledrejection', event => { - alert(event.reason); // Promise Failed! -}); +// doesn't run: error handled +window.addEventListener('unhandledrejection', event => alert(event.reason)); ``` -We create a rejected `promise` and do not handle the error. So we have the "unhandled rejection" event (printed in browser console too). - -We wouldn't have it if we added `.catch`, like this: +But if we forget to add `.catch`, then, after the microtask queue is empty, the engine triggers the event: ```js run let promise = Promise.reject(new Error("Promise Failed!")); -*!* -promise.catch(err => alert('caught')); -*/!* -// no error, all quiet +// Promise Failed! window.addEventListener('unhandledrejection', event => alert(event.reason)); ``` -Now let's say, we'll be catching the error, but after `setTimeout`: +What if we handle the error later? Like this: ```js run let promise = Promise.reject(new Error("Promise Failed!")); *!* -setTimeout(() => promise.catch(err => alert('caught'))); +setTimeout(() => promise.catch(err => alert('caught')), 1000); */!* // Error: Promise Failed! window.addEventListener('unhandledrejection', event => alert(event.reason)); ``` -Now the unhandled rejection appears again. Why? Because `unhandledrejection` triggers when the microtask queue is complete. The engine examines promises and, if any of them is in "rejected" state, then the event is generated. +Now, if we run it, we'll see `Promise Failed!` first and then `caught`. -In the example, the `.catch` added by `setTimeout` triggers too, of course it does, but later, after `unhandledrejection` has already occurred. +If we didn't know about the microtasks queue, we could wonder: "Why did `unhandledrejection` handler run? We did catch and handle the error!" -## Summary +But now we understand that `unhandledrejection` is generated when the microtask queue is complete: the engine examines promises and, if any of them is in the "rejected" state, then the event triggers. -- Promise handling is always asynchronous, as all promise actions pass through the internal "promise jobs" queue, also called "microtask queue" (v8 term). +In the example above, `.catch` added by `setTimeout` also triggers. But it does so later, after `unhandledrejection` has already occurred, so it doesn't change anything. - **So, `.then/catch/finally` are called after the current code is finished.** - - If we need to guarantee that a piece of code is executed after `.then/catch/finally`, it's best to add it into a chained `.then` call. - -- There's also a "macrotask queue" that keeps various events, network operation results, `setTimeout`-scheduled calls, and so on. These are also called "macrotasks" (v8 term). +## Summary - The engine uses the macrotask queue to handle them in the appearance order. +Promise handling is always asynchronous, as all promise actions pass through the internal "promise jobs" queue, also called "microtask queue" (V8 term). - **Macrotasks run after the code is finished *and* after the microtask queue is empty.** +So `.then/catch/finally` handlers are always called after the current code is finished. - In other words, they have lower priority. +If we need to guarantee that a piece of code is executed after `.then/catch/finally`, we can add it into a chained `.then` call. -So the order is: regular code, then promise handling, then everything else, like events etc. +In most Javascript engines, including browsers and Node.js, the concept of microtasks is closely tied with the "event loop" and "macrotasks". As these have no direct relation to promises, they are covered in another part of the tutorial, in the article <info:event-loop>. diff --git a/1-js/11-async/08-async-await/01-rewrite-async/solution.md b/1-js/11-async/08-async-await/01-rewrite-async/solution.md index c736defe9..3337ef3c4 100644 --- a/1-js/11-async/08-async-await/01-rewrite-async/solution.md +++ b/1-js/11-async/08-async-await/01-rewrite-async/solution.md @@ -13,13 +13,13 @@ async function loadJson(url) { // (1) throw new Error(response.status); } -loadJson('no-such-user.json') +loadJson('https://javascript.info/no-such-user.json') .catch(alert); // Error: 404 (4) ``` Notes: -1. The function `loadUrl` becomes `async`. +1. The function `loadJson` becomes `async`. 2. All `.then` inside are replaced with `await`. 3. We can `return response.json()` instead of awaiting for it, like this: diff --git a/1-js/11-async/08-async-await/01-rewrite-async/task.md b/1-js/11-async/08-async-await/01-rewrite-async/task.md index bf54e6f63..0c31737da 100644 --- a/1-js/11-async/08-async-await/01-rewrite-async/task.md +++ b/1-js/11-async/08-async-await/01-rewrite-async/task.md @@ -1,7 +1,7 @@ # Rewrite using async/await -Rewrite the one of examples from the chapter <info:promise-chaining> using `async/await` instead of `.then/catch`: +Rewrite this example code from the chapter <info:promise-chaining> using `async/await` instead of `.then/catch`: ```js run function loadJson(url) { @@ -12,9 +12,9 @@ function loadJson(url) { } else { throw new Error(response.status); } - }) + }); } -loadJson('no-such-user.json') // (3) +loadJson('https://javascript.info/no-such-user.json') .catch(alert); // Error: 404 ``` diff --git a/1-js/11-async/08-async-await/02-rewrite-async-2/solution.md b/1-js/11-async/08-async-await/02-rewrite-async-2/solution.md index 105948833..aa462dbf7 100644 --- a/1-js/11-async/08-async-await/02-rewrite-async-2/solution.md +++ b/1-js/11-async/08-async-await/02-rewrite-async-2/solution.md @@ -1,5 +1,5 @@ -There are no tricks here. Just replace `.catch` with `try...catch` inside `demoGithubUser` and add `async/await` where needed: +There are no tricks here. Just replace `.catch` with `try..catch` inside `demoGithubUser` and add `async/await` where needed: ```js run class HttpError extends Error { diff --git a/1-js/11-async/08-async-await/02-rewrite-async-2/task.md b/1-js/11-async/08-async-await/02-rewrite-async-2/task.md index 94af2e579..13d625d2a 100644 --- a/1-js/11-async/08-async-await/02-rewrite-async-2/task.md +++ b/1-js/11-async/08-async-await/02-rewrite-async-2/task.md @@ -1,7 +1,7 @@ -# Rewrite "rethrow" async/await +# Rewrite "rethrow" with async/await -Below you can find the "rethrow" example from the chapter <info:promise-chaining>. Rewrite it using `async/await` instead of `.then/catch`. +Below you can find the "rethrow" example. Rewrite it using `async/await` instead of `.then/catch`. And get rid of the recursion in favour of a loop in `demoGithubUser`: with `async/await` that becomes easy to do. @@ -22,7 +22,7 @@ function loadJson(url) { } else { throw new HttpError(response); } - }) + }); } // Ask for a user name until github returns a valid user diff --git a/1-js/11-async/08-async-await/03-async-from-regular/task.md b/1-js/11-async/08-async-await/03-async-from-regular/task.md index 18d0e2ce7..ca7c186ff 100644 --- a/1-js/11-async/08-async-await/03-async-from-regular/task.md +++ b/1-js/11-async/08-async-await/03-async-from-regular/task.md @@ -1,7 +1,7 @@ # Call async from non-async -We have a "regular" function. How to call `async` from it and use its result? +We have a "regular" function called `f`. How can you call the `async` function `wait()` and use its result inside of `f`? ```js async function wait() { @@ -11,7 +11,7 @@ async function wait() { } function f() { - // ...what to write here? + // ...what should you write here? // we need to call async wait() and wait to get 10 // remember, we can't use "await" } diff --git a/1-js/11-async/08-async-await/04-promise-all-failure/solution.md b/1-js/11-async/08-async-await/04-promise-all-failure/solution.md new file mode 100644 index 000000000..9fda8e000 --- /dev/null +++ b/1-js/11-async/08-async-await/04-promise-all-failure/solution.md @@ -0,0 +1,113 @@ + +The root of the problem is that `Promise.all` immediately rejects when one of its promises rejects, but it do nothing to cancel the other promises. + +In our case, the second query fails, so `Promise.all` rejects, and the `try...catch` block catches this error.Meanwhile, other promises are *not affected* - they independently continue their execution. In our case, the third query throws an error of its own after a bit of time. And that error is never caught, we can see it in the console. + +The problem is especially dangerous in server-side environments, such as Node.js, when an uncaught error may cause the process to crash. + +How to fix it? + +An ideal solution would be to cancel all unfinished queries when one of them fails. This way we avoid any potential errors. + +However, the bad news is that service calls (such as `database.query`) are often implemented by a 3rd-party library which doesn't support cancellation. Then there's no way to cancel a call. + +As an alternative, we can write our own wrapper function around `Promise.all` which adds a custom `then/catch` handler to each promise to track them: results are gathered and, if an error occurs, all subsequent promises are ignored. + +```js +function customPromiseAll(promises) { + return new Promise((resolve, reject) => { + const results = []; + let resultsCount = 0; + let hasError = false; // we'll set it to true upon first error + + promises.forEach((promise, index) => { + promise + .then(result => { + if (hasError) return; // ignore the promise if already errored + results[index] = result; + resultsCount++; + if (resultsCount === promises.length) { + resolve(results); // when all results are ready - successs + } + }) + .catch(error => { + if (hasError) return; // ignore the promise if already errored + hasError = true; // wops, error! + reject(error); // fail with rejection + }); + }); + }); +} +``` + +This approach has an issue of its own - it's often undesirable to `disconnect()` when queries are still in the process. + +It may be important that all queries complete, especially if some of them make important updates. + +So we should wait until all promises are settled before going further with the execution and eventually disconnecting. + +Here's another implementation. It behaves similar to `Promise.all` - also resolves with the first error, but waits until all promises are settled. + +```js +function customPromiseAllWait(promises) { + return new Promise((resolve, reject) => { + const results = new Array(promises.length); + let settledCount = 0; + let firstError = null; + + promises.forEach((promise, index) => { + Promise.resolve(promise) + .then(result => { + results[index] = result; + }) + .catch(error => { + if (firstError === null) { + firstError = error; + } + }) + .finally(() => { + settledCount++; + if (settledCount === promises.length) { + if (firstError !== null) { + reject(firstError); + } else { + resolve(results); + } + } + }); + }); + }); +} +``` + +Now `await customPromiseAllWait(...)` will stall the execution until all queries are processed. + +This is a more reliable approach, as it guarantees a predictable execution flow. + +Lastly, if we'd like to process all errors, we can use either use `Promise.allSettled` or write a wrapper around it to gathers all errors in a single [AggregateError](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/AggregateError) object and rejects with it. + +```js +// wait for all promises to settle +// return results if no errors +// throw AggregateError with all errors if any +function allOrAggregateError(promises) { + return Promise.allSettled(promises).then(results => { + const errors = []; + const values = []; + + results.forEach((res, i) => { + if (res.status === 'fulfilled') { + values[i] = res.value; + } else { + errors.push(res.reason); + } + }); + + if (errors.length > 0) { + throw new AggregateError(errors, 'One or more promises failed'); + } + + return values; + }); +} +``` diff --git a/1-js/11-async/08-async-await/04-promise-all-failure/task.md b/1-js/11-async/08-async-await/04-promise-all-failure/task.md new file mode 100644 index 000000000..74571c43e --- /dev/null +++ b/1-js/11-async/08-async-await/04-promise-all-failure/task.md @@ -0,0 +1,79 @@ + +# Dangerous Promise.all + +`Promise.all` is a great way to parallelize multiple operations. It's especially useful when we need to make parallel requests to multiple services. + +However, there's a hidden danger. We'll see an example in this task and explore how to avoid it. + +Let's say we have a connection to a remote service, such as a database. + +There're two functions: `connect()` and `disconnect()`. + +When connected, we can send requests using `database.query(...)` - an async function which usually returns the result but also may throw an error. + +Here's a simple implementation: + +```js +let database; + +function connect() { + database = { + async query(isOk) { + if (!isOk) throw new Error('Query failed'); + } + }; +} + +function disconnect() { + database = null; +} + +// intended usage: +// connect() +// ... +// database.query(true) to emulate a successful call +// database.query(false) to emulate a failed call +// ... +// disconnect() +``` + +Now here's the problem. + +We wrote the code to connect and send 3 queries in parallel (all of them take different time, e.g. 100, 200 and 300ms), then disconnect: + +```js +// Helper function to call async function `fn` after `ms` milliseconds +function delay(fn, ms) { + return new Promise((resolve, reject) => { + setTimeout(() => fn().then(resolve, reject), ms); + }); +} + +async function run() { + connect(); + + try { + await Promise.all([ + // these 3 parallel jobs take different time: 100, 200 and 300 ms + // we use the `delay` helper to achieve this effect +*!* + delay(() => database.query(true), 100), + delay(() => database.query(false), 200), + delay(() => database.query(false), 300) +*/!* + ]); + } catch(error) { + console.log('Error handled (or was it?)'); + } + + disconnect(); +} + +run(); +``` + +Two of these queries happen to be unsuccessful, but we're smart enough to wrap the `Promise.all` call into a `try..catch` block. + +However, this doesn't help! This script actually leads to an uncaught error in console! + +Why? How to avoid it? \ No newline at end of file diff --git a/1-js/11-async/08-async-await/article.md b/1-js/11-async/08-async-await/article.md index 041a5a465..e679b1c4c 100644 --- a/1-js/11-async/08-async-await/article.md +++ b/1-js/11-async/08-async-await/article.md @@ -12,9 +12,9 @@ async function f() { } ``` -The word "async" before a function means one simple thing: a function always returns a promise. Even If a function actually returns a non-promise value, prepending the function definition with the "async" keyword directs JavaScript to automatically wrap that value in a resolved promise. +The word "async" before a function means one simple thing: a function always returns a promise. Other values are wrapped in a resolved promise automatically. -For instance, the code above returns a resolved promise with the result of `1`, let's test it: +For instance, this function returns a resolved promise with the result of `1`; let's test it: ```js run async function f() { @@ -24,7 +24,7 @@ async function f() { f().then(alert); // 1 ``` -...We could explicitly return a promise, that would be the same as: +...We could explicitly return a promise, which would be the same: ```js run async function f() { @@ -56,7 +56,7 @@ async function f() { }); *!* - let result = await promise; // wait till the promise resolves (*) + let result = await promise; // wait until the promise resolves (*) */!* alert(result); // "done!" @@ -67,12 +67,12 @@ f(); The function execution "pauses" at the line `(*)` and resumes when the promise settles, with `result` becoming its result. So the code above shows "done!" in one second. -Let's emphasize: `await` literally makes JavaScript wait until the promise settles, and then go on with the result. That doesn't cost any CPU resources, because the engine can do other jobs meanwhile: execute other scripts, handle events etc. +Let's emphasize: `await` literally suspends the function execution until the promise settles, and then resumes it with the promise result. That doesn't cost any CPU resources, because the JavaScript engine can do other jobs in the meantime: execute other scripts, handle events, etc. -It's just a more elegant syntax of getting the promise result than `promise.then`, easier to read and write. +It's just a more elegant syntax of getting the promise result than `promise.then`. And, it's easier to read and write. ````warn header="Can't use `await` in regular functions" -If we try to use `await` in non-async function, there would be a syntax error: +If we try to use `await` in a non-async function, there would be a syntax error: ```js run function f() { @@ -83,7 +83,7 @@ function f() { } ``` -We will get this error if we do not put `async` before a function. As said, `await` only works inside an `async function`. +We may get this error if we forget to put `async` before a function. As stated earlier, `await` only works inside an `async` function. ```` Let's take the `showAvatar()` example from the chapter <info:promise-chaining> and rewrite it using `async/await`: @@ -121,18 +121,24 @@ showAvatar(); Pretty clean and easy to read, right? Much better than before. -````smart header="`await` won't work in the top-level code" -People who are just starting to use `await` tend to forget the fact that we can't use `await` in top-level code. For example, this will not work: +````smart header="Modern browsers allow top-level `await` in modules" +In modern browsers, `await` on top level works just fine, when we're inside a module. We'll cover modules in article <info:modules-intro>. -```js run -// syntax error in top-level code +For instance: + +```js run module +// we assume this code runs at top level, inside a module let response = await fetch('/article/promise-chaining/user.json'); let user = await response.json(); + +console.log(user); ``` -We can wrap it into an anonymous async function, like this: +If we're not using modules, or [older browsers](https://caniuse.com/mdn-javascript_operators_await_top_level) must be supported, there's a universal recipe: wrapping into an anonymous async function. -```js run +Like this: + +```js (async () => { let response = await fetch('/article/promise-chaining/user.json'); let user = await response.json(); @@ -140,12 +146,12 @@ We can wrap it into an anonymous async function, like this: })(); ``` - ```` + ````smart header="`await` accepts \"thenables\"" -Like `promise.then`, `await` allows to use thenable objects (those with a callable `then` method). The idea is that a 3rd-party object may not be a promise, but promise-compatible: if it supports `.then`, that's enough to use with `await`. +Like `promise.then`, `await` allows us to use thenable objects (those with a callable `then` method). The idea is that a third-party object may not be a promise, but promise-compatible: if it supports `.then`, that's enough to use it with `await`. -Here's a demo `Thenable` class, the `await` below accepts its instances: +Here's a demo `Thenable` class; the `await` below accepts its instances: ```js run class Thenable { @@ -157,7 +163,7 @@ class Thenable { // resolve with this.num*2 after 1000ms setTimeout(() => resolve(this.num * 2), 1000); // (*) } -}; +} async function f() { // waits for 1 second, then result becomes 2 @@ -168,10 +174,10 @@ async function f() { f(); ``` -If `await` gets a non-promise object with `.then`, it calls that method providing native functions `resolve`, `reject` as arguments. Then `await` waits until one of them is called (in the example above it happens in the line `(*)`) and then proceeds with the result. +If `await` gets a non-promise object with `.then`, it calls that method providing the built-in functions `resolve` and `reject` as arguments (just as it does for a regular `Promise` executor). Then `await` waits until one of them is called (in the example above it happens in the line `(*)`) and then proceeds with the result. ```` -````smart header="Async methods" +````smart header="Async class methods" To declare an async class method, just prepend it with `async`: ```js run @@ -185,14 +191,14 @@ class Waiter { new Waiter() .wait() - .then(alert); // 1 + .then(alert); // 1 (this is the same as (result => alert(result))) ``` The meaning is the same: it ensures that the returned value is a promise and enables `await`. ```` ## Error handling -If a promise resolves normally, then `await promise` returns the result. But in case of a rejection, it throws the error, just as if there were a `throw` statement at that line. +If a promise resolves normally, then `await promise` returns the result. But in the case of a rejection, it throws the error, just as if there were a `throw` statement at that line. This code: @@ -204,7 +210,7 @@ async function f() { } ``` -...Is the same as this: +...is the same as this: ```js async function f() { @@ -214,7 +220,7 @@ async function f() { } ``` -In real situations, the promise may take some time before it rejects. So `await` will wait, and then throw an error. +In real situations, the promise may take some time before it rejects. In that case there will be a delay before `await` throws an error. We can catch that error using `try..catch`, the same way as a regular `throw`: @@ -233,7 +239,7 @@ async function f() { f(); ``` -In case of an error, the control jumps to the `catch` block. We can also wrap multiple lines: +In the case of an error, the control jumps to the `catch` block. We can also wrap multiple lines: ```js run async function f() { @@ -263,15 +269,13 @@ f().catch(alert); // TypeError: failed to fetch // (*) */!* ``` -If we forget to add `.catch` there, then we get an unhandled promise error (viewable in the console). We can catch such errors using a global event handler as described in the chapter <info:promise-error-handling>. +If we forget to add `.catch` there, then we get an unhandled promise error (viewable in the console). We can catch such errors using a global `unhandledrejection` event handler as described in the chapter <info:promise-error-handling>. ```smart header="`async/await` and `promise.then/catch`" -When we use `async/await`, we rarely need `.then`, because `await` handles the waiting for us. And we can use a regular `try..catch` instead of `.catch`. That's usually (not always) more convenient. +When we use `async/await`, we rarely need `.then`, because `await` handles the waiting for us. And we can use a regular `try..catch` instead of `.catch`. That's usually (but not always) more convenient. -But at the top level of the code, when we're outside of any `async` function, we're syntactically unable to use `await`, so it's a normal practice to add `.then/catch` to handle the final result or falling-through errors. - -Like in the line `(*)` of the example above. +But at the top level of the code, when we're outside any `async` function, we're syntactically unable to use `await`, so it's a normal practice to add `.then/catch` to handle the final result or falling-through error, like in the line `(*)` of the example above. ``` ````smart header="`async/await` works well with `Promise.all`" @@ -286,50 +290,22 @@ let results = await Promise.all([ ]); ``` -In case of an error, it propagates as usual: from the failed promise to `Promise.all`, and then becomes an exception that we can catch using `try..catch` around the call. +In the case of an error, it propagates as usual, from the failed promise to `Promise.all`, and then becomes an exception that we can catch using `try..catch` around the call. ```` -## Microtask queue [#microtask-queue] - -As we've seen in the chapter <info:microtask-queue>, promise handlers are executed asynchronously. Every `.then/catch/finally` handler first gets into the "microtask queue" and executed after the current code is complete. - -`Async/await` is based on promises, so it uses the same microtask queue internally, and has the similar priority over macrotasks. - -For instance, we have: -- `setTimeout(handler, 0)`, that should run `handler` with zero delay. -- `let x = await f()`, function `f()` is async, but returns immediately. - -Which one runs first if `await` is *below* `setTimeout` in the code? - -```js run -async function f() { - return 1; -} - -(async () => { - setTimeout(() => alert('timeout'), 0); - - await f(); - - alert('await'); -})(); -``` - -There's no ambiguity here: `await` always finishes first, because (as a microtask) it has a higher priority than `setTimeout` handling. - ## Summary The `async` keyword before a function has two effects: 1. Makes it always return a promise. -2. Allows to use `await` in it. +2. Allows `await` to be used in it. The `await` keyword before a promise makes JavaScript wait until that promise settles, and then: -1. If it's an error, the exception is generated, same as if `throw error` were called at that very place. -2. Otherwise, it returns the result, so we can assign it to a value. +1. If it's an error, an exception is generated — same as if `throw error` were called at that very place. +2. Otherwise, it returns the result. -Together they provide a great framework to write asynchronous code that is easy both to read and write. +Together they provide a great framework to write asynchronous code that is easy to both read and write. -With `async/await` we rarely need to write `promise.then/catch`, but we still shouldn't forget that they are based on promises, because sometimes (e.g. in the outermost scope) we have to use these methods. Also `Promise.all` is a nice thing to wait for many tasks simultaneously. +With `async/await` we rarely need to write `promise.then/catch`, but we still shouldn't forget that they are based on promises, because sometimes (e.g. in the outermost scope) we have to use these methods. Also `Promise.all` is nice when we are waiting for many tasks simultaneously. diff --git a/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/solution.md b/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/solution.md index 8bb521506..4355d0cfc 100644 --- a/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/solution.md +++ b/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/solution.md @@ -3,7 +3,7 @@ function* pseudoRandom(seed) { let value = seed; while(true) { - value = value * 16807 % 2147483647 + value = value * 16807 % 2147483647; yield value; } @@ -35,4 +35,4 @@ alert(generator()); // 282475249 alert(generator()); // 1622650073 ``` -That's fine for this context. But then we loose ability to iterate with `for..of` and to use generator composition, that may be useful elsewhere. +That also works. But then we lose ability to iterate with `for..of` and to use generator composition, that may be useful elsewhere. diff --git a/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/task.md b/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/task.md index c2ac53181..e7c251ad3 100644 --- a/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/task.md +++ b/1-js/12-generators-iterators/1-generators/01-pseudo-random-generator/task.md @@ -3,11 +3,11 @@ There are many areas where we need random data. -One of them is testing. We may need random data: text, numbers etc, to test things out well. +One of them is testing. We may need random data: text, numbers, etc. to test things out well. In JavaScript, we could use `Math.random()`. But if something goes wrong, we'd like to be able to repeat the test, using exactly the same data. -For that, so called "seeded pseudo-random generators" are used. They take a "seed", the first value, and then generate next ones using a formula. So that the same seed yields the same sequence, and hence the whole flow is easily reproducible. We only need to remember the seed to repeat it. +For that, so called "seeded pseudo-random generators" are used. They take a "seed", the first value, and then generate the next ones using a formula so that the same seed yields the same sequence, and hence the whole flow is easily reproducible. We only need to remember the seed to repeat it. An example of such formula, that generates somewhat uniformly distributed values: diff --git a/1-js/12-generators-iterators/1-generators/article.md b/1-js/12-generators-iterators/1-generators/article.md index f6fcfb00d..1f83a7e0c 100644 --- a/1-js/12-generators-iterators/1-generators/article.md +++ b/1-js/12-generators-iterators/1-generators/article.md @@ -1,9 +1,8 @@ - # Generators Regular functions return only one, single value (or nothing). -Generators can return ("yield") multiple values, possibly an infinite number of values, one after another, on-demand. They work great with [iterables](info:iterable), allowing to create data streams with ease. +Generators can return ("yield") multiple values, one after another, on-demand. They work great with [iterables](info:iterable), allowing to create data streams with ease. ## Generator functions @@ -19,20 +18,33 @@ function* generateSequence() { } ``` -When `generateSequence()` is called, it does not execute the code. Instead, it returns a special object, called "generator". +Generator functions behave differently from regular ones. When such function is called, it doesn't run its code. Instead it returns a special object, called "generator object", to manage the execution. + +Here, take a look: + +```js run +function* generateSequence() { + yield 1; + yield 2; + return 3; +} -```js // "generator function" creates "generator object" let generator = generateSequence(); +*!* +alert(generator); // [object Generator] +*/!* ``` -The `generator` object can be perceived as a "frozen function call": +The function code execution hasn't started yet:  -Upon creation, the code execution is paused at the very beginning. +The main method of a generator is `next()`. When called, it runs the execution until the nearest `yield <value>` statement (`value` can be omitted, then it's `undefined`). Then the function execution pauses, and the yielded `value` is returned to the outer code. -The main method of a generator is `next()`. When called, it resumes execution till the nearest `yield <value>` statement. Then the execution pauses, and the value is returned to the outer code. +The result of `next()` is always an object with two properties: +- `value`: the yielded value. +- `done`: `true` if the function code has finished, otherwise `false`. For instance, here we create the generator and get its first yielded value: @@ -52,15 +64,17 @@ let one = generator.next(); alert(JSON.stringify(one)); // {value: 1, done: false} ``` -The result of `next()` is always an object: -- `value`: the yielded value. -- `done`: `false` if the code is not finished yet, otherwise `true`. +As of now, we got the first value only, and the function execution is on the second line: -As of now, we got the first value only: + +<<<<<<< HEAD  Let's call `generator.next()` again. It resumes the execution and returns the next `yield`: +======= +Let's call `generator.next()` again. It resumes the code execution and returns the next `yield`: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js let two = generator.next(); @@ -70,7 +84,7 @@ alert(JSON.stringify(two)); // {value: 2, done: false}  -And, if we call it the third time, then the execution reaches `return` statement that finishes the function: +And, if we call it a third time, the execution reaches the `return` statement that finishes the function: ```js let three = generator.next(); @@ -82,14 +96,10 @@ alert(JSON.stringify(three)); // {value: 3, *!*done: true*/!*} Now the generator is done. We should see it from `done:true` and process `value:3` as the final result. -New calls `generator.next()` don't make sense any more. If we make them, they return the same object: `{done: true}`. - -There's no way to "roll back" a generator. But we can create another one by calling `generateSequence()`. - -So far, the most important thing to understand is that generator functions, unlike regular function, do not run the code. They serve as "generator factories". Running `function*` returns a generator, and then we ask it for values. +New calls to `generator.next()` don't make sense any more. If we do them, they return the same object: `{done: true}`. ```smart header="`function* f(…)` or `function *f(…)`?" -That's a minor religious question, both syntaxes are correct. +Both syntaxes are correct. But usually the first syntax is preferred, as the star `*` denotes that it's a generator function, it describes the kind, not the name, so it should stick with the `function` keyword. ``` @@ -98,7 +108,7 @@ But usually the first syntax is preferred, as the star `*` denotes that it's a g As you probably already guessed looking at the `next()` method, generators are [iterable](info:iterable). -We can get loop over values by `for..of`: +We can loop over their values using `for..of`: ```js run function* generateSequence() { @@ -114,11 +124,11 @@ for(let value of generator) { } ``` -That's a much better-looking way to work with generators than calling `.next().value`, right? +Looks a lot nicer than calling `.next().value`, right? ...But please note: the example above shows `1`, then `2`, and that's all. It doesn't show `3`! -It's because for-of iteration ignores the last `value`, when `done: true`. So, if we want all results to be shown by `for..of`, we must return them with `yield`: +It's because `for..of` iteration ignores the last `value`, when `done: true`. So, if we want all results to be shown by `for..of`, we must return them with `yield`: ```js run function* generateSequence() { @@ -136,7 +146,7 @@ for(let value of generator) { } ``` -Naturally, as generators are iterable, we can call all related functionality, e.g. the spread operator `...`: +As generators are iterable, we can call all related functionality, e.g. the spread syntax `...`: ```js run function* generateSequence() { @@ -150,9 +160,9 @@ let sequence = [0, ...generateSequence()]; alert(sequence); // 0, 1, 2, 3 ``` -In the code above, `...generateSequence()` turns the iterable into array of items (read more about the spread operator in the chapter [](info:rest-parameters-spread-operator#spread-operator)) +In the code above, `...generateSequence()` turns the iterable generator object into an array of items (read more about the spread syntax in the chapter [](info:rest-parameters-spread#spread-syntax)) -## Using generators instead of iterables +## Using generators for iterables Some time ago, in the chapter [](info:iterable) we created an iterable `range` object that returns values `from..to`. @@ -163,7 +173,7 @@ let range = { from: 1, to: 5, - // for..of calls this method once in the very beginning + // for..of range calls this method once in the very beginning [Symbol.iterator]() { // ...it returns the iterator object: // onward, for..of works only with that object, asking it for next values @@ -184,28 +194,13 @@ let range = { } }; +// iteration over range returns numbers from range.from to range.to alert([...range]); // 1,2,3,4,5 ``` -Using a generator to make iterable sequences is so much more elegant: - -```js run -function* generateSequence(start, end) { - for (let i = start; i <= end; i++) { - yield i; - } -} - -let sequence = [...generateSequence(1,5)]; - -alert(sequence); // 1, 2, 3, 4, 5 -``` - -...But what if we'd like to keep a custom `range` object? +We can use a generator function for iteration by providing it as `Symbol.iterator`. -## Converting Symbol.iterator to generator - -We can get the best from both worlds by providing a generator as `Symbol.iterator`: +Here's the same `range`, but much more compact: ```js run let range = { @@ -222,39 +217,44 @@ let range = { alert( [...range] ); // 1,2,3,4,5 ``` -The `range` object is now iterable. - -That works pretty well, because when `range[Symbol.iterator]` is called: -- it returns an object (now a generator) -- that has `.next()` method (yep, a generator has it) -- that returns values in the form `{value: ..., done: true/false}` (check, exactly what generator does). +That works, because `range[Symbol.iterator]()` now returns a generator, and generator methods are exactly what `for..of` expects: +- it has a `.next()` method +- that returns values in the form `{value: ..., done: true/false}` -That's not a coincidence, of course. Generators aim to make iterables easier, so we can see that. +That's not a coincidence, of course. Generators were added to JavaScript language with iterators in mind, to implement them easily. -The last variant with a generator is much more concise than the original iterable code, and keeps the same functionality. +The variant with a generator is much more concise than the original iterable code of `range`, and keeps the same functionality. -```smart header="Generators may continue forever" +```smart header="Generators may generate values forever" In the examples above we generated finite sequences, but we can also make a generator that yields values forever. For instance, an unending sequence of pseudo-random numbers. -That surely would require a `break` in `for..of`, otherwise the loop would repeat forever and hang. +That surely would require a `break` (or `return`) in `for..of` over such generator. Otherwise, the loop would repeat forever and hang. ``` ## Generator composition Generator composition is a special feature of generators that allows to transparently "embed" generators in each other. -For instance, we'd like to generate a sequence of: -- digits `0..9` (character codes 48..57), -- followed by alphabet letters `a..z` (character codes 65..90) -- followed by uppercased letters `A..Z` (character codes 97..122) +For instance, we have a function that generates a sequence of numbers: -Then we plan to create passwords by selecting characters from it (could add syntax characters as well), but need to generate the sequence first. +```js +function* generateSequence(start, end) { + for (let i = start; i <= end; i++) yield i; +} +``` -We already have `function* generateSequence(start, end)`. Let's reuse it to deliver 3 sequences one after another, together they are exactly what we need. +Now we'd like to reuse it to generate a more complex sequence: +- first, digits `0..9` (with character codes 48..57), +- followed by uppercase alphabet letters `A..Z` (character codes 65..90) +- followed by lowercase alphabet letters `a..z` (character codes 97..122) + +We can use this sequence e.g. to create passwords by selecting characters from it (could add syntax characters as well), but let's generate it first. In a regular function, to combine results from multiple other functions, we call them, store the results, and then join at the end. -For generators, we can do better, like this: +For generators, there's a special `yield*` syntax to "embed" (compose) one generator into another. + +The composed generator: ```js run function* generateSequence(start, end) { @@ -285,7 +285,7 @@ for(let code of generatePasswordCodes()) { alert(str); // 0..9A..Za..z ``` -The special `yield*` directive in the example is responsible for the composition. It *delegates* the execution to another generator. Or, to say it simple, it runs generators and transparently forwards their yields outside, as if they were done by the calling generator itself. +The `yield*` directive *delegates* the execution to another generator. This term means that `yield* gen` iterates over the generator `gen` and transparently forwards its yields outside. As if the values were yielded by the outer generator. The result is the same as if we inlined the code from nested generators: @@ -318,17 +318,13 @@ for(let code of generateAlphaNum()) { alert(str); // 0..9A..Za..z ``` -A generator composition is a natural way to insert a flow of one generator into another. - -It works even if the flow of values from the nested generator is infinite. It's simple and doesn't use extra memory to store intermediate results. +A generator composition is a natural way to insert a flow of one generator into another. It doesn't use extra memory to store intermediate results. -## "yield" is a two-way road +## "yield" is a two-way street -Till this moment, generators were like "iterators on steroids". And that's how they are often used. +Until this moment, generators were similar to iterable objects, with a special syntax to generate values. But in fact they are much more powerful and flexible. -But in fact they are much more powerful and flexible. - -That's because `yield` is a two-way road: it not only returns the result outside, but also can pass the value inside the generator. +That's because `yield` is a two-way street: it not only returns the result to the outside, but also can pass the value inside the generator. To do so, we should call `generator.next(arg)`, with an argument. That argument becomes the result of `yield`. @@ -338,7 +334,7 @@ Let's see an example: function* gen() { *!* // Pass a question to the outer code and wait for an answer - let result = yield "2 + 2?"; // (*) + let result = yield "2 + 2 = ?"; // (*) */!* alert(result); @@ -353,37 +349,39 @@ generator.next(4); // --> pass the result into the generator  -1. The first call `generator.next()` is always without an argument. It starts the execution and returns the result of the first `yield` ("2+2?"). At this point the generator pauses the execution (still on that line). +1. The first call `generator.next()` should be always made without an argument (the argument is ignored if passed). It starts the execution and returns the result of the first `yield "2+2=?"`. At this point the generator pauses the execution, while staying on the line `(*)`. 2. Then, as shown at the picture above, the result of `yield` gets into the `question` variable in the calling code. 3. On `generator.next(4)`, the generator resumes, and `4` gets in as the result: `let result = 4`. -Please note, the outer code does not have to immediately call`next(4)`. It may take time to calculate the value. This is also a valid code: +Please note, the outer code does not have to immediately call `next(4)`. It may take time. That's not a problem: the generator will wait. + +For instance: ```js // resume the generator after some time setTimeout(() => generator.next(4), 1000); ``` -The syntax may seem a bit odd. It's quite uncommon for a function and the calling code to pass values around to each other. But that's exactly what's going on. +As we can see, unlike regular functions, a generator and the calling code can exchange results by passing values in `next/yield`. To make things more obvious, here's another example, with more calls: ```js run function* gen() { - let ask1 = yield "2 + 2?"; + let ask1 = yield "2 + 2 = ?"; alert(ask1); // 4 - let ask2 = yield "3 * 3?" + let ask2 = yield "3 * 3 = ?" alert(ask2); // 9 } let generator = gen(); -alert( generator.next().value ); // "2 + 2?" +alert( generator.next().value ); // "2 + 2 = ?" -alert( generator.next(4).value ); // "3 * 3?" +alert( generator.next(4).value ); // "3 * 3 = ?" alert( generator.next(9).done ); // true ``` @@ -408,12 +406,12 @@ As we observed in the examples above, the outer code may pass a value into the g To pass an error into a `yield`, we should call `generator.throw(err)`. In that case, the `err` is thrown in the line with that `yield`. -For instance, here the yield of `"2 + 2?"` leads to an error: +For instance, here the yield of `"2 + 2 = ?"` leads to an error: ```js run function* gen() { try { - let result = yield "2 + 2?"; // (1) + let result = yield "2 + 2 = ?"; // (1) alert("The execution does not reach here, because the exception is thrown above"); } catch(e) { @@ -430,7 +428,7 @@ generator.throw(new Error("The answer is not found in my database")); // (2) */!* ``` -The error, thrown into the generator at the line `(2)` leads to an exception in the line `(1)` with `yield`. In the example above, `try..catch` catches it and shows. +The error, thrown into the generator at line `(2)` leads to an exception in line `(1)` with `yield`. In the example above, `try..catch` catches it and shows it. If we don't catch it, then just like any exception, it "falls out" the generator into the calling code. @@ -438,7 +436,7 @@ The current line of the calling code is the line with `generator.throw`, labelle ```js run function* generate() { - let result = yield "2 + 2?"; // Error in this line + let result = yield "2 + 2 = ?"; // Error in this line } let generator = generate(); @@ -456,14 +454,36 @@ try { If we don't catch the error there, then, as usual, it falls through to the outer calling code (if any) and, if uncaught, kills the script. +## generator.return + +`generator.return(value)` finishes the generator execution and return the given `value`. + +```js +function* gen() { + yield 1; + yield 2; + yield 3; +} + +const g = gen(); + +g.next(); // { value: 1, done: false } +g.return('foo'); // { value: "foo", done: true } +g.next(); // { value: undefined, done: true } +``` + +If we again use `generator.return()` in a completed generator, it will return that value again ([MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Generator/return)). + +Often we don't use it, as most of time we want to get all returning values, but it can be useful when we want to stop generator in a specific condition. + ## Summary -- Generators are created by generator functions `function*(…) {…}`. +- Generators are created by generator functions `function* f(…) {…}`. - Inside generators (only) there exists a `yield` operator. - The outer code and the generator may exchange results via `next/yield` calls. -In modern JavaScript, generators are rarely used. But sometimes they come in handy, because the ability of a function to exchange data with the calling code during the execution is quite unique. +In modern JavaScript, generators are rarely used. But sometimes they come in handy, because the ability of a function to exchange data with the calling code during the execution is quite unique. And, surely, they are great for making iterable objects. -Also, in the next chapter we'll learn async generators, which are used to read streams of asynchronously generated data in `for` loop. +Also, in the next chapter we'll learn async generators, which are used to read streams of asynchronously generated data (e.g paginated fetches over a network) in `for await ... of` loops. -In web-programming we often work with streamed data, e.g. need to fetch paginated results, so that's a very important use case. +In web-programming we often work with streamed data, so that's another very important use case. diff --git a/1-js/12-generators-iterators/1-generators/generateSequence-2.svg b/1-js/12-generators-iterators/1-generators/generateSequence-2.svg index 4c64e983e..b598c6d30 100644 --- a/1-js/12-generators-iterators/1-generators/generateSequence-2.svg +++ b/1-js/12-generators-iterators/1-generators/generateSequence-2.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-2.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 26.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M286.5 42.5v6h32v2h-32v6l-14-7 14-7z"/><text id="{value:-1,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="54">{value: 1, done: false}</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-2.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 26.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M286.5 42.5v6h32v2h-32v6l-14-7 14-7z"/><text id="{value:-1,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="54">{value: 1, done: false}</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-2.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 26.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M286.5 42.5v6h32v2h-32v6l-14-7 14-7z" transform="matrix(-1 0 0 1 590 0)"/><text id="{value:-1,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="54">{value: 1, done: false}</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/12-generators-iterators/1-generators/generateSequence-3.svg b/1-js/12-generators-iterators/1-generators/generateSequence-3.svg index 0af8e9efd..720c88d52 100644 --- a/1-js/12-generators-iterators/1-generators/generateSequence-3.svg +++ b/1-js/12-generators-iterators/1-generators/generateSequence-3.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-3.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 25.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M287.5 57.5v6h32v2h-32v6l-14-7 14-7z"/><text id="{value:-2,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="69">{value: 2, done: false}</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-3.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 25.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M287.5 57.5v6h32v2h-32v6l-14-7 14-7z"/><text id="{value:-2,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="69">{value: 2, done: false}</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="519" height="150" viewBox="0 0 519 150"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="generator" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="generateSequence-3.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M26 17h271v110H26z"/><g id="function*-generateSe" fill-rule="nonzero" transform="translate(40.12 25.99)"><path id="function" fill="#1C85B5" d="M.112 10.01v-.938h1.666V3.948H.112V3.01h1.666v-.392c0-.877.229-1.512.686-1.904.457-.392 1.11-.588 1.96-.588.308 0 .583.019.826.056.243.037.504.112.784.224l-.252.91a2.862 2.862 0 00-.7-.203 4.295 4.295 0 00-.616-.049c-.616 0-1.036.152-1.26.455-.224.303-.336.8-.336 1.491h2.898v.938H2.87v5.124h2.898v.938H.112zm12.25-7h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H13.51l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08H7.63V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm4.62 2.45a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm2.688-1.078h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm9.436 7v-.938h2.436V3.948H42.35V3.01h3.556v6.062h2.38v.938H42.35zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm5.67 5.488c0-1.13.294-2.023.882-2.681.588-.658 1.428-.987 2.52-.987.588 0 1.094.096 1.519.287.425.191.777.45 1.057.777.28.327.488.714.623 1.162.135.448.203.929.203 1.442 0 .56-.075 1.066-.224 1.519-.15.453-.369.838-.658 1.155-.29.317-.646.562-1.071.735a3.82 3.82 0 01-1.449.259c-.579 0-1.083-.096-1.512-.287a2.967 2.967 0 01-1.064-.777 3.168 3.168 0 01-.623-1.162 4.963 4.963 0 01-.203-1.442zm1.162 0c0 .327.04.653.119.98.08.327.208.62.385.882.177.261.408.471.693.63.285.159.632.238 1.043.238.747 0 1.309-.231 1.687-.693.378-.462.567-1.141.567-2.037 0-.336-.04-.665-.119-.987a2.608 2.608 0 00-.392-.875 2.066 2.066 0 00-.7-.63c-.285-.159-.632-.238-1.043-.238-.747 0-1.307.229-1.68.686-.373.457-.56 1.139-.56 2.044zm7.742-1.05a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46z"/><path id="*" fill="#DBAF88" d="M69.062.014l1.246 2.45L71.512 0l.966.518-1.694 2.366 2.842-.21v1.134l-2.758-.224 1.624 2.254-.938.546-1.302-2.506-1.232 2.506-1.008-.546 1.694-2.254-2.772.224V2.674l2.772.238L68.096.56z"/><path id="generateSequence" fill="#181717" d="M90.216 10.36c0 .439-.084.824-.252 1.155a2.269 2.269 0 01-.693.819 3.127 3.127 0 01-1.029.483 4.81 4.81 0 01-1.274.161c-.55 0-1.031-.056-1.442-.168a4.29 4.29 0 01-1.064-.434l.532-.994c.084.075.194.147.329.217s.292.135.469.196c.177.06.366.11.567.147.2.037.399.056.595.056.41 0 .754-.037 1.029-.112.275-.075.497-.191.665-.35.168-.159.29-.364.364-.616a3.29 3.29 0 00.112-.924v-.784h-.056a2.18 2.18 0 01-.826.714c-.336.168-.77.252-1.302.252-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.579 0 1.071.04 1.477.119.406.08.777.18 1.113.301v7.042zm-3.122-1.12c.57 0 1.017-.147 1.344-.441.327-.294.555-.744.686-1.351V4.13c-.41-.196-.985-.294-1.722-.294-.747 0-1.335.22-1.764.658-.43.439-.644 1.11-.644 2.016 0 .401.037.768.112 1.099.075.331.194.618.357.861.163.243.378.432.644.567.266.135.595.203.987.203zm11.592-.112c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H93.31c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm14.504 3.668c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.872 6.244v-.938h1.89V3.948h-1.89V3.01h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.988-6.454a4.447 4.447 0 011.358-.504 7.423 7.423 0 011.484-.154c.476 0 .866.075 1.169.224.303.15.541.343.714.581.173.238.29.509.35.812.06.303.091.609.091.917 0 .355-.01.733-.028 1.134-.019.401-.033.803-.042 1.204 0 .467.028.91.084 1.33h.938v.91h-1.862l-.126-1.05h-.07c-.056.084-.14.191-.252.322-.112.13-.259.259-.441.385a2.749 2.749 0 01-1.589.469c-.69 0-1.237-.177-1.638-.532-.401-.355-.602-.84-.602-1.456 0-.476.105-.873.315-1.19.21-.317.511-.56.903-.728.392-.168.866-.266 1.421-.294a9.544 9.544 0 011.869.112c.047-.43.054-.786.021-1.071-.033-.285-.107-.511-.224-.679a.938.938 0 00-.49-.357 2.482 2.482 0 00-.777-.105c-.42 0-.821.058-1.204.175-.383.117-.723.236-1.022.357l-.35-.812zm2.058 5.642c.261 0 .504-.042.728-.126a2.22 2.22 0 00.588-.322 2.133 2.133 0 00.672-.868v-.98a7.92 7.92 0 00-1.344-.126c-.41 0-.765.044-1.064.133-.299.089-.532.226-.7.413-.168.187-.252.43-.252.728 0 .308.105.576.315.805.21.229.562.343 1.057.343zm5.264-6.188h1.666V1.624l1.092-.308V3.01h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253V3.948h-1.666V3.01zm15.372 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm6.048 4.858c.28.15.595.273.945.371.35.098.702.147 1.057.147.308 0 .597-.03.868-.091.27-.06.506-.159.707-.294.2-.135.36-.306.476-.511a1.45 1.45 0 00.175-.728c0-.383-.128-.695-.385-.938a3.824 3.824 0 00-.959-.644 17.73 17.73 0 00-1.246-.546 7.417 7.417 0 01-1.246-.63 3.498 3.498 0 01-.959-.889c-.257-.35-.385-.796-.385-1.337 0-.383.077-.728.231-1.036a2.24 2.24 0 01.651-.784c.28-.215.618-.38 1.015-.497a4.68 4.68 0 011.323-.175c.56 0 1.069.04 1.526.119.457.08.821.184 1.092.315l.028-.014V.49h.014l-.014.042v2.24h-1.036V1.274a7.635 7.635 0 00-.77-.14 6.788 6.788 0 00-.896-.056c-.308 0-.586.037-.833.112a2.08 2.08 0 00-.63.301 1.37 1.37 0 00-.399.441 1.08 1.08 0 00-.14.532c0 .364.128.667.385.91s.576.46.959.651.798.38 1.246.567c.448.187.863.404 1.246.651s.702.544.959.889.385.775.385 1.288c0 .42-.077.798-.231 1.134a2.494 2.494 0 01-.658.868 2.97 2.97 0 01-1.05.56c-.415.13-.884.196-1.407.196-.355 0-.69-.028-1.008-.084a6.883 6.883 0 01-1.589-.476 7.642 7.642 0 01-.497-.238l.014-.042V7.07h1.036v1.554zm13.706.504c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm11.158 9.044h-1.092V9.226h-.056c-.215.308-.49.544-.826.707-.336.163-.77.245-1.302.245-1.064 0-1.846-.296-2.345-.889-.5-.593-.749-1.514-.749-2.765 0-1.195.317-2.098.952-2.709.635-.611 1.577-.917 2.828-.917.55 0 1.064.047 1.54.14.476.093.826.191 1.05.294v9.478zm-3.108-3.57c1.11 0 1.783-.583 2.016-1.75V4.116a2.435 2.435 0 00-.735-.217 6.723 6.723 0 00-.973-.063c-.747 0-1.335.215-1.764.644-.43.43-.644 1.106-.644 2.03 0 .401.035.768.105 1.099.07.331.189.618.357.861.168.243.385.432.651.567.266.135.595.203.987.203zm9.38-6.23h1.988v4.774c0 .457.028.896.084 1.316h.938v.91h-1.862l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938V3.01h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19V3.948h-.896V3.01zm10.724 6.118c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.25 1.694a14.051 14.051 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469A2.91 2.91 0 01196 2.954c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092V6.202c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078V5.46zm13.244-1.372a5.531 5.531 0 00-.693-.189 3.61 3.61 0 00-.735-.077c-.915 0-1.596.215-2.044.644-.448.43-.672 1.11-.672 2.044 0 .41.065.777.196 1.099.13.322.317.595.56.819.243.224.534.397.875.518.34.121.716.182 1.127.182.439 0 .866-.075 1.281-.224.415-.15.763-.345 1.043-.588l.49.812c-.13.112-.294.229-.49.35a4.837 4.837 0 01-1.533.602 4.962 4.962 0 01-1.015.098c-.607 0-1.141-.086-1.603-.259a3.045 3.045 0 01-1.155-.742 3.176 3.176 0 01-.7-1.162 4.525 4.525 0 01-.238-1.505c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.69 0 1.253.06 1.687.182.434.121.8.257 1.099.406l-.014.042v1.946h-1.008v-1.33zm9.66 5.04c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939h-5.418c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511z"/><path id="(){" fill="#7E7C7B" d="M223.328 12.978a6.964 6.964 0 01-1.757-.707 5.036 5.036 0 01-1.428-1.225c-.406-.504-.728-1.12-.966-1.848s-.357-1.591-.357-2.59c0-.99.121-1.855.364-2.597s.57-1.372.98-1.89c.41-.518.889-.933 1.435-1.246a6.005 6.005 0 011.729-.665l.35.882a7.086 7.086 0 00-1.512.623 3.928 3.928 0 00-1.183 1.022c-.331.425-.588.952-.77 1.582-.182.63-.273 1.393-.273 2.289 0 .905.11 1.678.329 2.317.22.64.509 1.171.868 1.596.36.425.76.756 1.204.994.443.238.889.404 1.337.497l-.35.966zm4.354-.966a4.965 4.965 0 001.337-.497 4.15 4.15 0 001.204-.994c.36-.425.649-.957.868-1.596.22-.64.329-1.412.329-2.317 0-.896-.091-1.659-.273-2.289-.182-.63-.439-1.157-.77-1.582a3.928 3.928 0 00-1.183-1.022 7.086 7.086 0 00-1.512-.623l.35-.882c.607.13 1.183.352 1.729.665a4.984 4.984 0 011.435 1.246c.41.518.737 1.148.98 1.89s.364 1.608.364 2.597c0 .999-.119 1.862-.357 2.59-.238.728-.56 1.344-.966 1.848a5.036 5.036 0 01-1.428 1.225 6.964 6.964 0 01-1.757.707l-.35-.966zm17.878-2.45c0-.401-.068-.749-.203-1.043a2.44 2.44 0 00-.504-.735c-.2-.196-.415-.34-.644-.434a1.641 1.641 0 00-.609-.14v-.98c.177 0 .38-.04.609-.119.229-.08.443-.21.644-.392.2-.182.369-.418.504-.707.135-.29.203-.649.203-1.078V2.506c0-.336.054-.644.161-.924a2.22 2.22 0 01.448-.728c.191-.205.42-.366.686-.483.266-.117.558-.175.875-.175h2.03v.98h-1.54c-.588 0-.996.14-1.225.42-.229.28-.343.695-.343 1.246V4.2c0 .383-.07.723-.21 1.022a2.7 2.7 0 01-1.106 1.225c-.205.117-.373.184-.504.203v.084c.121.019.285.091.49.217.205.126.406.299.602.518.196.22.366.478.511.777.145.299.217.63.217.994v1.344c0 .57.121.99.364 1.26s.649.406 1.218.406h1.526v.98h-2.03c-.653 0-1.178-.194-1.575-.581-.397-.387-.595-.959-.595-1.715V9.562z"/><path id="yield" fill="#1C85B5" d="M20.034 25.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="1" fill="#A7333A" d="M67.858 26.002h2.198v-7.364l-2.24 1.568-.546-.798 3.08-2.198h.784v8.792h2.156v1.008h-5.432z"/><path id=";" fill="#7E7C7B" d="M77.686 26.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="yield" fill="#1C85B5" d="M20.034 42.932h.532l1.736-5.922h1.134l-1.946 6.286c-.103.327-.208.698-.315 1.113a5.2 5.2 0 01-.455 1.169 2.807 2.807 0 01-.798.924c-.336.252-.775.378-1.316.378-.121 0-.25-.014-.385-.042a3.856 3.856 0 01-.399-.105 2.574 2.574 0 01-.364-.147.977.977 0 01-.252-.168l.392-1.008c.056.047.133.093.231.14.098.047.205.086.322.119.117.033.231.06.343.084.112.023.205.035.28.035.7 0 1.176-.593 1.428-1.778h-.938l-2.94-7h1.26l2.45 5.922zm5.516 1.078v-.938h2.436v-5.124H25.55v-.938h3.556v6.062h2.38v.938H25.55zm2.058-8.988c0-.252.084-.469.252-.651a.838.838 0 01.644-.273c.27 0 .497.091.679.273a.889.889 0 01.273.651.806.806 0 01-.273.616.963.963 0 01-.679.252.872.872 0 01-.644-.252.838.838 0 01-.252-.616zm12.278 8.106c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H34.51c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm5.614-3.556h2.324v7.448c0 .57.096.97.287 1.204.191.233.474.35.847.35.261 0 .511-.047.749-.14.238-.093.502-.252.791-.476l.504.77c-.15.13-.313.247-.49.35a3.638 3.638 0 01-1.106.42 2.963 2.963 0 01-1.442-.077 1.579 1.579 0 01-.679-.427 1.897 1.897 0 01-.413-.777c-.093-.322-.14-.721-.14-1.197v-6.51h-1.232v-.938zm11.62 0h2.268v7.406c0 .084.005.189.014.315a15.6 15.6 0 00.084.812c.019.135.037.254.056.357h.938v.91h-1.862l-.14-1.078h-.056c-.205.355-.511.651-.917.889s-.875.357-1.407.357c-1.055 0-1.832-.296-2.331-.889-.5-.593-.749-1.514-.749-2.765 0-.588.084-1.108.252-1.561.168-.453.408-.83.721-1.134a3.118 3.118 0 011.127-.693 4.298 4.298 0 011.47-.238c.196 0 .371.005.525.014.154.01.296.023.427.042.13.019.254.044.371.077.117.033.245.068.385.105v-1.988h-1.176v-.938zm-.84 9.03c.57 0 1.017-.147 1.344-.441.327-.294.55-.735.672-1.323v-3.248a2.168 2.168 0 00-.693-.308c-.257-.065-.595-.098-1.015-.098-.747 0-1.335.217-1.764.651-.43.434-.644 1.118-.644 2.051 0 .383.035.74.105 1.071.07.331.187.618.35.861.163.243.38.434.651.574.27.14.602.21.994.21z"/><path id="2" fill="#A7333A" d="M73.08 36.688c0 .859-.406 1.815-1.218 2.87-.812 1.055-1.946 2.203-3.402 3.444h4.928v1.008h-6.216v-1.008c.177-.168.42-.387.728-.658.308-.27.64-.572.994-.903a29.01 29.01 0 001.085-1.071c.369-.383.702-.775 1.001-1.176.299-.401.541-.803.728-1.204a2.75 2.75 0 00.28-1.162c0-.56-.147-1.001-.441-1.323-.294-.322-.735-.483-1.323-.483-.504 0-.929.056-1.274.168a3.101 3.101 0 00-.938.49l-.476-.77c.42-.299.866-.518 1.337-.658s.987-.21 1.547-.21c.877 0 1.54.238 1.988.714.448.476.672 1.12.672 1.932z"/><path id=";" fill="#7E7C7B" d="M77.686 43.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672z"/><path id="return" fill="#1C85B5" d="M16.604 61.01v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm14.882-.882c-.168.15-.366.29-.595.42a4.39 4.39 0 01-.749.336c-.27.093-.553.166-.847.217-.294.051-.59.077-.889.077-.56 0-1.055-.086-1.484-.259a2.858 2.858 0 01-1.078-.742 3.234 3.234 0 01-.658-1.155 4.767 4.767 0 01-.224-1.512c0-.588.082-1.11.245-1.568.163-.457.399-.84.707-1.148a3.08 3.08 0 011.12-.707 4.187 4.187 0 011.47-.245c.392 0 .78.051 1.162.154.383.103.721.296 1.015.581.294.285.52.677.679 1.176.159.5.215 1.146.168 1.939H26.11c0 .84.226 1.468.679 1.883.453.415 1.057.623 1.813.623.252 0 .502-.03.749-.091.247-.06.483-.133.707-.217a3.72 3.72 0 00.595-.28c.173-.103.306-.196.399-.28l.434.798zm-2.954-5.362c-.308 0-.6.033-.875.098a1.995 1.995 0 00-.728.329c-.21.154-.383.355-.518.602s-.222.553-.259.917h4.354c-.047-.616-.245-1.094-.595-1.435-.35-.34-.81-.511-1.379-.511zm4.382-.756h1.666v-1.386l1.092-.308v1.694h3.738v.938h-3.738v3.304c0 .681.166 1.185.497 1.512.331.327.805.49 1.421.49.42 0 .786-.08 1.099-.238.313-.159.595-.331.847-.518l.364.826a3.938 3.938 0 01-1.169.63 4.182 4.182 0 01-1.393.238 3.41 3.41 0 01-1.057-.161 2.426 2.426 0 01-.875-.497 2.38 2.38 0 01-.602-.861c-.15-.35-.224-.768-.224-1.253v-3.472h-1.666v-.938zm13.048 0h1.988v4.774c0 .457.028.896.084 1.316h.938v.91H47.11l-.042-1.19h-.056a2.938 2.938 0 01-.966.98c-.401.252-.887.378-1.456.378-.43 0-.796-.049-1.099-.147a1.556 1.556 0 01-.749-.511c-.196-.243-.34-.567-.434-.973-.093-.406-.14-.912-.14-1.519v-3.08h-.938v-.938h2.03v3.794c0 .383.016.726.049 1.029.033.303.1.56.203.77.103.21.254.369.455.476.2.107.469.161.805.161.485 0 .908-.159 1.267-.476.36-.317.632-.714.819-1.19v-3.626h-.896v-.938zm4.242 7v-.938h1.89v-5.124h-1.89v-.938h2.618l.224.91h.07c.317-.29.672-.537 1.064-.742.392-.205.859-.308 1.4-.308.317 0 .572.06.763.182.191.121.34.299.448.532.107.233.18.516.217.847.037.331.056.707.056 1.127l-.98.014c0-.597-.058-1.036-.175-1.316-.117-.28-.329-.42-.637-.42-.29 0-.55.042-.784.126a2.565 2.565 0 00-.609.308c-.173.121-.32.243-.441.364s-.21.224-.266.308v4.13h2.688v.938h-5.656zm8.778-4.55a14.05 14.05 0 00-.063-1.176 4.657 4.657 0 00-.049-.364h-.938v-.91h1.876l.126 1.134h.07c.093-.159.217-.315.371-.469a2.91 2.91 0 011.225-.721c.252-.075.518-.112.798-.112.43 0 .81.047 1.141.14.331.093.607.259.826.497.22.238.385.56.497.966.112.406.168.922.168 1.547v4.018h-1.092v-3.808c0-.775-.126-1.358-.378-1.75-.252-.392-.71-.588-1.372-.588-.243 0-.478.049-.707.147a2.44 2.44 0 00-.623.385c-.187.159-.35.343-.49.553-.14.21-.243.432-.308.665v4.396h-1.078v-4.55z"/><path id="3" fill="#A7333A" d="M77.952 60.212c.373 0 .716-.049 1.029-.147a2.65 2.65 0 00.819-.413 1.9 1.9 0 00.546-.644c.13-.252.196-.532.196-.84 0-.644-.226-1.104-.679-1.379-.453-.275-1.062-.413-1.827-.413h-.896v-.672l2.758-3.486h-4.186V51.21h5.53v1.008l-2.758 3.234h.224c.43 0 .821.06 1.176.182.355.121.663.294.924.518.261.224.464.495.609.812.145.317.217.677.217 1.078 0 .504-.098.952-.294 1.344a2.89 2.89 0 01-.798.987c-.336.266-.73.467-1.183.602a5.034 5.034 0 01-1.449.203c-.476 0-.889-.035-1.239-.105-.35-.07-.67-.156-.959-.259l.28-.98c.243.112.527.203.854.273a5.3 5.3 0 001.106.105z"/><path id=";}" fill="#7E7C7B" d="M86.086 60.24c0-.28.091-.506.273-.679.182-.173.418-.259.707-.259.355 0 .63.124.826.371.196.247.294.595.294 1.043a2.36 2.36 0 01-.637 1.659 2.465 2.465 0 01-.588.462c-.21.117-.404.203-.581.259l-.35-.546c.336-.13.625-.317.868-.56.243-.243.364-.537.364-.882-.13.028-.233.042-.308.042-.27 0-.483-.077-.637-.231-.154-.154-.231-.38-.231-.679zm.126-5.502c0-.299.084-.53.252-.693.168-.163.392-.245.672-.245.299 0 .532.082.7.245.168.163.252.394.252.693 0 .27-.084.495-.252.672-.168.177-.401.266-.7.266-.28 0-.504-.089-.672-.266a.938.938 0 01-.252-.672zM4.2 78.934c0 .756-.198 1.328-.595 1.715-.397.387-.922.581-1.575.581H0v-.98h1.526c.57 0 .975-.135 1.218-.406.243-.27.364-.69.364-1.26V77.24c0-.364.07-.695.21-.994.14-.299.308-.558.504-.777.196-.22.397-.392.602-.518.205-.126.373-.198.504-.217v-.084c-.13-.019-.299-.086-.504-.203a2.7 2.7 0 01-1.106-1.225c-.14-.299-.21-.64-.21-1.022v-1.358c0-.55-.114-.966-.343-1.246-.229-.28-.637-.42-1.225-.42H0v-.98h2.03c.317 0 .609.058.875.175.266.117.495.278.686.483.191.205.34.448.448.728.107.28.161.588.161.924v1.428c0 .43.068.789.203 1.078.135.29.303.525.504.707.2.182.415.313.644.392.229.08.432.119.609.119v.98c-.177 0-.38.047-.609.14a2.011 2.011 0 00-.644.434 2.44 2.44 0 00-.504.735c-.135.294-.203.642-.203 1.043v1.372z"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M287.5 57.5v6h32v2h-32v6l-14-7 14-7z" transform="matrix(-1 0 0 1 592 0)"/><text id="{value:-2,-done:-fal" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="326" y="69">{value: 2, done: false}</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/12-generators-iterators/2-async-iterators-generators/article.md b/1-js/12-generators-iterators/2-async-iterators-generators/article.md index 43b214a0f..d4e9f7861 100644 --- a/1-js/12-generators-iterators/2-async-iterators-generators/article.md +++ b/1-js/12-generators-iterators/2-async-iterators-generators/article.md @@ -1,35 +1,48 @@ -# Async iterators and generators +# Async iteration and generators -Asynchronous iterators allow to iterate over data that comes asynchronously, on-demand. +Asynchronous iteration allow us to iterate over data that comes asynchronously, on-demand. Like, for instance, when we download something chunk-by-chunk over a network. And asynchronous generators make it even more convenient. -For instance, when we download something chunk-by-chunk, or just expect events to come asynchronously and would like to iterate over them -- async iterators and generators may come in handy. Let's see a simple example first, to grasp the syntax, and then review a real-life use case. +Let's see a simple example first, to grasp the syntax, and then review a real-life use case. -## Async iterators +## Recall iterables -Asynchronous iterators are totally similar to regular iterators, with a few syntactic differences. +Let's recall the topic about iterables. -"Regular" iterable object from the chapter <info:iterable> look like this: +The idea is that we have an object, such as `range` here: +```js +let range = { + from: 1, + to: 5 +}; +``` + +...And we'd like to use `for..of` loop on it, such as `for(value of range)`, to get values from `1` to `5`. + +In other words, we want to add an *iteration ability* to the object. + +That can be implemented using a special method with the name `Symbol.iterator`: + +- This method is called in by the `for..of` construct when the loop is started, and it should return an object with the `next` method. +- For each iteration, the `next()` method is invoked for the next value. +- The `next()` should return a value in the form `{done: true/false, value:<loop value>}`, where `done:true` means the end of the loop. + +Here's an implementation for the iterable `range`: ```js run let range = { from: 1, to: 5, - // for..of calls this method once in the very beginning *!* - [Symbol.iterator]() { + [Symbol.iterator]() { // called once, in the beginning of for..of */!* - // ...it returns the iterator object: - // onward, for..of works only with that object, asking it for next values return { current: this.from, last: this.to, - // next() is called on each iteration by the for..of loop *!* - next() { // (2) - // it should return the value as an object {done:.., value :...} + next() { // called every iteration, to get the next value */!* if (this.current <= this.last) { return { done: false, value: this.current++ }; @@ -46,39 +59,46 @@ for(let value of range) { } ``` -If necessary, please refer to the [chapter about iterables](info:iterable) for details about regular iterators. +If anything is unclear, please visit the chapter [](info:iterable), it gives all the details about regular iterables. + +## Async iterables + +Asynchronous iteration is needed when values come asynchronously: after `setTimeout` or another kind of delay. -To make the object iterable asynchronously: -1. We need to use `Symbol.asyncIterator` instead of `Symbol.iterator`. -2. `next()` should return a promise. -3. To iterate over such an object, we should use `for await (let item of iterable)` loop. +The most common case is that the object needs to make a network request to deliver the next value, we'll see a real-life example of it a bit later. -Let's make an iterable `range` object, like the one before, but now it will return values asynchronously, one per second: +To make an object iterable asynchronously: + +1. Use `Symbol.asyncIterator` instead of `Symbol.iterator`. +2. The `next()` method should return a promise (to be fulfilled with the next value). + - The `async` keyword handles it, we can simply make `async next()`. +3. To iterate over such an object, we should use a `for await (let item of iterable)` loop. + - Note the `await` word. + +As a starting example, let's make an iterable `range` object, similar like the one before, but now it will return values asynchronously, one per second. + +All we need to do is to perform a few replacements in the code above: ```js run let range = { from: 1, to: 5, - // for await..of calls this method once in the very beginning *!* [Symbol.asyncIterator]() { // (1) */!* - // ...it returns the iterator object: - // onward, for await..of works only with that object, asking it for next values return { current: this.from, last: this.to, - // next() is called on each iteration by the for..of loop *!* async next() { // (2) - // it should return the value as an object {done:.., value :...} - // (automatically wrapped into a promise by async) */!* - // can use await inside, do async stuff: +*!* + // note: we can use "await" inside the async next: await new Promise(resolve => setTimeout(resolve, 1000)); // (3) +*/!* if (this.current <= this.last) { return { done: false, value: this.current++ }; @@ -101,38 +121,43 @@ let range = { })() ``` -As we can see, the components are similar to regular iterators: +As we can see, the structure is similar to regular iterators: 1. To make an object asynchronously iterable, it must have a method `Symbol.asyncIterator` `(1)`. -2. It must return the object with `next()` method returning a promise `(2)`. -3. The `next()` method doesn't have to be `async`, it may be a regular method returning a promise, but `async` allows to use `await` inside. Here we just delay for a second `(3)`. +2. This method must return the object with `next()` method returning a promise `(2)`. +3. The `next()` method doesn't have to be `async`, it may be a regular method returning a promise, but `async` allows us to use `await`, so that's convenient. Here we just delay for a second `(3)`. 4. To iterate, we use `for await(let value of range)` `(4)`, namely add "await" after "for". It calls `range[Symbol.asyncIterator]()` once, and then its `next()` for values. -Here's a small cheatsheet: +Here's a small table with the differences: | | Iterators | Async iterators | |-------|-----------|-----------------| -| Object method to provide iteraterable | `Symbol.iterator` | `Symbol.asyncIterator` | +| Object method to provide iterator | `Symbol.iterator` | `Symbol.asyncIterator` | | `next()` return value is | any value | `Promise` | | to loop, use | `for..of` | `for await..of` | - -````warn header="The spread operator doesn't work asynchronously" +````warn header="The spread syntax `...` doesn't work asynchronously" Features that require regular, synchronous iterators, don't work with asynchronous ones. -For instance, a spread operator won't work: +For instance, a spread syntax won't work: ```js alert( [...range] ); // Error, no Symbol.iterator ``` -That's natural, as it expects to find `Symbol.iterator`, same as `for..of` without `await`. +That's natural, as it expects to find `Symbol.iterator`, not `Symbol.asyncIterator`. + +It's also the case for `for..of`: the syntax without `await` needs `Symbol.iterator`. ```` -## Async generators +## Recall generators -JavaScript also provides generators, that are also iterable. +Now let's recall generators, as they allow to make iteration code much shorter. Most of the time, when we'd like to make an iterable, we'll use generators. -Let's recall a sequence generator from the chapter [](info:generators). It generates a sequence of values from `start` to `end` (could be anything else): +For sheer simplicity, omitting some important stuff, they are "functions that generate (yield) values". They are explained in detail in the chapter [](info:generators). + +Generators are labelled with `function*` (note the star) and use `yield` to generate a value, then we can use `for..of` to loop over them. + +This example generates a sequence of values from `start` to `end`: ```js run function* generateSequence(start, end) { @@ -146,12 +171,54 @@ for(let value of generateSequence(1, 5)) { } ``` +As we already know, to make an object iterable, we should add `Symbol.iterator` to it. + +```js +let range = { + from: 1, + to: 5, +*!* + [Symbol.iterator]() { + return <object with next to make range iterable> + } +*/!* +} +``` + +A common practice for `Symbol.iterator` is to return a generator, it makes the code shorter, as you can see: + +```js run +let range = { + from: 1, + to: 5, + + *[Symbol.iterator]() { // a shorthand for [Symbol.iterator]: function*() + for(let value = this.from; value <= this.to; value++) { + yield value; + } + } +}; + +for(let value of range) { + alert(value); // 1, then 2, then 3, then 4, then 5 +} +``` -Normally, we can't use `await` in generators. All values must come synchronously: there's no place for delay in `for..of`. +Please see the chapter [](info:generators) if you'd like more details. -But what if we need to use `await` in the generator body? To perform network requests, for instance. +In regular generators we can't use `await`. All values must come synchronously, as required by the `for..of` construct. -No problem, just prepend it with `async`, like this: +What if we'd like to generate values asynchronously? From network requests, for instance. + +Let's switch to asynchronous generators to make it possible. + +## Async generators (finally) + +For most practical applications, when we'd like to make an object that asynchronously generates a sequence of values, we can use an asynchronous generator. + +The syntax is simple: prepend `function*` with `async`. That makes the generator asynchronous. + +And then use `for await (...)` to iterate over it, like this: ```js run *!*async*/!* function* generateSequence(start, end) { @@ -159,7 +226,7 @@ No problem, just prepend it with `async`, like this: for (let i = start; i <= end; i++) { *!* - // yay, can use await! + // Wow, can use await! await new Promise(resolve => setTimeout(resolve, 1000)); */!* @@ -172,70 +239,43 @@ No problem, just prepend it with `async`, like this: let generator = generateSequence(1, 5); for *!*await*/!* (let value of generator) { - alert(value); // 1, then 2, then 3, then 4, then 5 + alert(value); // 1, then 2, then 3, then 4, then 5 (with delay between) } })(); ``` -Now we have an the async generator, iteratable with `for await...of`. +As the generator is asynchronous, we can use `await` inside it, rely on promises, perform network requests and so on. -It's indeed very simple. We add the `async` keyword, and the generator now can use `await` inside of it, rely on promises and other async functions. +````smart header="Under-the-hood difference" +Technically, if you're an advanced reader who remembers the details about generators, there's an internal difference. -Technically, another the difference of an async generator is that its `generator.next()` method is now asynchronous also, it returns promises. +For async generators, the `generator.next()` method is asynchronous, it returns promises. -Instead of `result = generator.next()` for a regular, non-async generator, values can be obtained like this: +In a regular generator we'd use `result = generator.next()` to get values. In an async generator, we should add `await`, like this: ```js result = await generator.next(); // result = {value: ..., done: true/false} ``` +That's why async generators work with `for await...of`. +```` -## Iterables via async generators - -When we'd like to make an object iterable, we should add `Symbol.iterator` to it. - -```js -let range = { - from: 1, - to: 5, -*!* - [Symbol.iterator]() { ...return object with next to make range iterable... } -*/!* -} -``` - -A common practice for `Symbol.iterator` is to return a generator, rather than a plain object with `next` as in the example before. - -Let's recall an example from the chapter [](info:generators): - -```js run -let range = { - from: 1, - to: 5, - - *[Symbol.iterator]() { // a shorthand for [Symbol.iterator]: function*() - for(let value = this.from; value <= this.to; value++) { - yield value; - } - } -}; +### Async iterable range -for(let value of range) { - alert(value); // 1, then 2, then 3, then 4, then 5 -} -``` +Regular generators can be used as `Symbol.iterator` to make the iteration code shorter. -Here a custom object `range` is iterable, and the generator `*[Symbol.iterator]` implements the logic for listing values. +Similar to that, async generators can be used as `Symbol.asyncIterator` to implement the asynchronous iteration. -If we'd like to add async actions into the generator, then we should replace `Symbol.iterator` with async `Symbol.asyncIterator`: +For instance, we can make the `range` object generate values asynchronously, once per second, by replacing synchronous `Symbol.iterator` with asynchronous `Symbol.asyncIterator`: ```js run let range = { from: 1, to: 5, + // this line is same as [Symbol.asyncIterator]: async function*() { *!* - async *[Symbol.asyncIterator]() { // same as [Symbol.asyncIterator]: async function*() + async *[Symbol.asyncIterator]() { */!* for(let value = this.from; value <= this.to; value++) { @@ -258,31 +298,39 @@ let range = { Now values come with a delay of 1 second between them. -## Real-life example +```smart +Technically, we can add both `Symbol.iterator` and `Symbol.asyncIterator` to the object, so it's both synchronously (`for..of`) and asynchronously (`for await..of`) iterable. + +In practice though, that would be a weird thing to do. +``` + +## Real-life example: paginated data -So far we've seen simple examples, to gain basic understanding. Now let's review a real-life use case. +So far we've seen basic examples, to gain understanding. Now let's review a real-life use case. -There are many online APIs that deliver paginated data. For instance, when we need a list of users, then we can fetch it page-by-page: a request returns a pre-defined count (e.g. 100 users), and provides an URL to the next page. +There are many online services that deliver paginated data. For instance, when we need a list of users, a request returns a pre-defined count (e.g. 100 users) - "one page", and provides a URL to the next page. -The pattern is very common, it's not about users, but just about anything. For instance, GitHub allows to retrieve commits in the same, paginated fashion: +This pattern is very common. It's not about users, but just about anything. -- We should make a request to URL in the form `https://api.github.com/repos/<repo>/commits`. +For instance, GitHub allows us to retrieve commits in the same, paginated fashion: + +- We should make a request to `fetch` in the form `https://api.github.com/repos/<repo>/commits`. - It responds with a JSON of 30 commits, and also provides a link to the next page in the `Link` header. - Then we can use that link for the next request, to get more commits, and so on. -What we'd like to have is an iterable source of commits, so that we could use it like this: +For our code, we'd like to have a simpler way to get commits. -```js -let repo = 'javascript-tutorial/en.javascript.info'; // GitHub repository to get commits from +Let's make a function `fetchCommits(repo)` that gets commits for us, making requests whenever needed. And let it care about all pagination stuff. For us it'll be a simple async iteration `for await..of`. + +So the usage will be like this: -for await (let commit of fetchCommits(repo)) { +```js +for await (let commit of fetchCommits("username/repository")) { // process commit } ``` -We'd like `fetchCommits` to get commits for us, making requests whenever needed. And let it care about all pagination stuff, for us it'll be a simple `for await..of`. - -With async generators that's pretty easy to implement: +Here's such function, implemented as async generator: ```js async function* fetchCommits(repo) { @@ -290,14 +338,14 @@ async function* fetchCommits(repo) { while (url) { const response = await fetch(url, { // (1) - headers: {'User-Agent': 'Our script'}, // github requires user-agent header + headers: {'User-Agent': 'Our script'}, // github needs any user-agent header }); - const body = await response.json(); // (2) parses response as JSON (array of commits) + const body = await response.json(); // (2) response is JSON (array of commits) // (3) the URL of the next page is in the headers, extract it let nextPage = response.headers.get('Link').match(/<(.*?)>; rel="next"/); - nextPage = nextPage && nextPage[1]; + nextPage = nextPage?.[1]; url = nextPage; @@ -308,10 +356,16 @@ async function* fetchCommits(repo) { } ``` -1. We use the browser `fetch` method to download from a remote URL. It allows to supply authorization and other headers if needed, here GitHub requires `User-Agent`. -2. The fetch result is parsed as JSON, that's again a `fetch`-specific method. -3. We can get the next page URL from the `Link` header of the response. It has a special format, so we use a regexp for that. The next page URL may look like this: `https://api.github.com/repositories/93253246/commits?page=2`, it's generated by GitHub itself. -4. Then we yield all commits received, and when they finish -- the next `while(url)` iteration will trigger, making one more request. +More explanations about how it works: + +1. We use the browser [fetch](info:fetch) method to download the commits. + + - The initial URL is `https://api.github.com/repos/<repo>/commits`, and the next page will be in the `Link` header of the response. + - The `fetch` method allows us to supply authorization and other headers if needed -- here GitHub requires `User-Agent`. +2. The commits are returned in JSON format. +3. We should get the next page URL from the `Link` header of the response. It has a special format, so we use a regular expression for that (we will learn this feature in [Regular expressions](info:regular-expressions)). + - The next page URL may look like `https://api.github.com/repositories/93253246/commits?page=2`. It's generated by GitHub itself. +4. Then we yield the received commits one by one, and when they finish, the next `while(url)` iteration will trigger, making one more request. An example of use (shows commit authors in console): @@ -330,9 +384,13 @@ An example of use (shows commit authors in console): } })(); + +// Note: If you are running this in an external sandbox, you'll need to paste here the function fetchCommits described above ``` -That's just what we wanted. The internal pagination mechanics is invisible from the outside. For us it's just an async generator that returns commits. +That's just what we wanted. + +The internal mechanics of paginated requests is invisible from the outside. For us it's just an async generator that returns commits. ## Summary @@ -342,20 +400,18 @@ When we expect the data to come asynchronously, with delays, their async counter Syntax differences between async and regular iterators: -| | Iterators | Async iterators | +| | Iterable | Async Iterable | |-------|-----------|-----------------| -| Object method to provide iteraterable | `Symbol.iterator` | `Symbol.asyncIterator` | -| `next()` return value is | any value | `Promise` | +| Method to provide iterator | `Symbol.iterator` | `Symbol.asyncIterator` | +| `next()` return value is | `{value:…, done: true/false}` | `Promise` that resolves to `{value:…, done: true/false}` | Syntax differences between async and regular generators: | | Generators | Async generators | |-------|-----------|-----------------| | Declaration | `function*` | `async function*` | -| `generator.next()` returns | `{value:…, done: true/false}` | `Promise` that resolves to `{value:…, done: true/false}` | +| `next()` return value is | `{value:…, done: true/false}` | `Promise` that resolves to `{value:…, done: true/false}` | In web-development we often meet streams of data, when it flows chunk-by-chunk. For instance, downloading or uploading a big file. -We could use async generators to process such data, but there's also another API called Streams, that may be more convenient, as it provides special interfaces to transform the data and to pass it from one stream to another (e.g. download from one place and immediately send elsewhere). But they are also more complex. - -Streams API not a part of JavaScript language standard. Streams and async generators complement each other, both are great ways to handle async data flows. +We can use async generators to process such data. It's also noteworthy that in some environments, like in browsers, there's also another API called Streams, that provides special interfaces to work with such streams, to transform the data and to pass it from one stream to another (e.g. download from one place and immediately send elsewhere). diff --git a/1-js/12-generators-iterators/2-async-iterators-generators/head.html b/1-js/12-generators-iterators/2-async-iterators-generators/head.html index 74d66a8b8..03f21e2bd 100644 --- a/1-js/12-generators-iterators/2-async-iterators-generators/head.html +++ b/1-js/12-generators-iterators/2-async-iterators-generators/head.html @@ -11,7 +11,7 @@ // the URL of the next page is in the headers, extract it let nextPage = response.headers.get('Link').match(/<(.*?)>; rel="next"/); - nextPage = nextPage && nextPage[1]; + nextPage = nextPage?.[1]; url = nextPage; diff --git a/1-js/13-modules/01-modules-intro/article.md b/1-js/13-modules/01-modules-intro/article.md index ad4f21068..5ad70d151 100644 --- a/1-js/13-modules/01-modules-intro/article.md +++ b/1-js/13-modules/01-modules-intro/article.md @@ -1,29 +1,30 @@ # Modules, introduction -As our application grows bigger, we want to split it into multiple files, so called 'modules'. -A module usually contains a class or a library of useful functions. +As our application grows bigger, we want to split it into multiple files, so called "modules". A module may contain a class or a library of functions for a specific purpose. -For a long time, JavaScript existed without a language-level module syntax. That wasn't a problem, because initially scripts were small and simple. So there was no need. +For a long time, JavaScript existed without a language-level module syntax. That wasn't a problem, because initially scripts were small and simple, so there was no need. -But eventually scripts became more and more complex, so the community invented a variety of ways to organize code into modules. +But eventually scripts became more and more complex, so the community invented a variety of ways to organize code into modules, special libraries to load modules on demand. -For instance: +To name some (for historical reasons): -- [AMD](https://en.wikipedia.org/wiki/Asynchronous_module_definition) -- one of the most ancient module systems, initially implemented by the library [require.js](http://requirejs.org/). -- [CommonJS](http://wiki.commonjs.org/wiki/Modules/1.1) -- the module system created for Node.js server. +- [AMD](https://en.wikipedia.org/wiki/Asynchronous_module_definition) -- one of the most ancient module systems, initially implemented by the library [require.js](https://requirejs.org/). +- [CommonJS](https://wiki.commonjs.org/wiki/Modules/1.1) -- the module system created for Node.js server. - [UMD](https://github.com/umdjs/umd) -- one more module system, suggested as a universal one, compatible with AMD and CommonJS. -Now all these slowly become a part of history, but we still can find them in old scripts. The language-level module system appeared in the standard in 2015, gradually evolved since then, and is now supported by all major browsers and in Node.js. +Now these all slowly became a part of history, but we still can find them in old scripts. + +The language-level module system appeared in the standard in 2015, gradually evolved since then, and is now supported by all major browsers and in Node.js. So we'll study the modern JavaScript modules from now on. ## What is a module? -A module is just a file, a single script, as simple as that. +A module is just a file. One script is one module. As simple as that. -Directives `export` and `import` allow to interchange functionality between modules: +Modules can load each other and use special directives `export` and `import` to interchange functionality, call functions of one module from another one: -- `export` keyword labels variables and functions that should be accessible from outside the file. -- `import` allows to import functionality from other modules. +- `export` keyword labels variables and functions that should be accessible from outside the current module. +- `import` allows the import of functionality from other modules. For instance, if we have a file `sayHi.js` exporting a function: @@ -44,13 +45,21 @@ alert(sayHi); // function... sayHi('John'); // Hello, John! ``` -In this tutorial we concentrate on the language itself, but we use browser as the demo environment, so let's see how modules work in the browser. +The `import` directive loads the module by path `./sayHi.js` relative to the current file, and assigns exported function `sayHi` to the corresponding variable. + +Let's run the example in-browser. -To use modules, we must set the attribute `<script type="module">`, like this: +As modules support special keywords and features, we must tell the browser that a script should be treated as a module, by using the attribute `<script type="module">`. + +Like this: [codetabs src="say" height="140" current="index.html"] -The browser automatically fetches and evaluates imports, then runs the script. +The browser automatically fetches and evaluates the imported module (and its imports if needed), and then runs the script. + +```warn header="Modules work only via HTTP(s), not locally" +If you try to open a web-page locally, via `file://` protocol, you'll find that `import/export` directives don't work. Use a local web-server, such as [static-server](https://www.npmjs.com/package/static-server#getting-started) or use the "live server" capability of your editor, such as VS Code [Live Server Extension](https://marketplace.visualstudio.com/items?itemName=ritwickdey.LiveServer) to test modules. +``` ## Core module features @@ -60,7 +69,7 @@ There are core features, valid both for browser and server-side JavaScript. ### Always "use strict" -Modules always `use strict`. E.g. assigning to an undeclared variable will give an error. +Modules always work in strict mode. E.g. assigning to an undeclared variable will give an error. ```html run <script type="module"> @@ -68,25 +77,28 @@ Modules always `use strict`. E.g. assigning to an undeclared variable will give </script> ``` -Here we can see it in the browser, but the same is true for any module. - ### Module-level scope Each module has its own top-level scope. In other words, top-level variables and functions from a module are not seen in other scripts. -In the example below, two scripts are imported, and `hello.js` tries to use `user` variable declared in `user.js`, and fails: +In the example below, two scripts are imported, and `hello.js` tries to use `user` variable declared in `user.js`. It fails, because it's a separate module (you'll see the error in the console): [codetabs src="scopes" height="140" current="index.html"] -Modules are expected to `export` what they want to be accessible from outside and `import` what they need. +Modules should `export` what they want to be accessible from outside and `import` what they need. + +- `user.js` should export the `user` variable. +- `hello.js` should import it from `user.js` module. -So we should import `user.js` directly into `hello.js` instead of `index.html`. +In other words, with modules we use import/export instead of relying on global variables. -That's the correct variant: +This is the correct variant: [codetabs src="scopes-working" height="140" current="hello.js"] -In the browser, independant top-level scope also exists for each `<script type="module">`: +In the browser, if we talk about HTML pages, independent top-level scope also exists for each `<script type="module">`. + +Here are two scripts on the same page, both `type="module"`. They don't see each other's top-level variables: ```html run <script type="module"> @@ -101,13 +113,21 @@ In the browser, independant top-level scope also exists for each `<script type=" </script> ``` -If we really need to make a "global" in-browser variable, we can explicitly assign it to `window` and access as `window.user`. But that's an exception requiring a good reason. +```smart +In the browser, we can make a variable window-level global by explicitly assigning it to a `window` property, e.g. `window.user = "John"`. + +Then all scripts will see it, both with `type="module"` and without it. + +That said, making such global variables is frowned upon. Please try to avoid them. +``` ### A module code is evaluated only the first time when imported -If the same module is imported into multiple other places, its code is executed only the first time, then exports are given to all importers. +If the same module is imported into multiple other modules, its code is executed only once, upon the first import. Then its exports are given to all further importers. + +The one-time evaluation has important consequences, that we should be aware of. -That has important consequences. Let's see that on examples. +Let's see a couple of examples. First, if executing a module code brings side-effects, like showing a message, then importing it multiple times will trigger it only once -- the first time: @@ -123,12 +143,14 @@ alert("Module is evaluated!"); import `./alert.js`; // Module is evaluated! // 📁 2.js -import `./alert.js`; // (nothing) +import `./alert.js`; // (shows nothing) ``` -In practice, top-level module code is mostly used for initialization. We create data structures, pre-fill them, and if we want something to be reusable -- export it. +The second import shows nothing, because the module has already been evaluated. -Now, a more advanced example. +There's a rule: top-level module code should be used for initialization, creation of module-specific internal data structures. If we need to make something callable multiple times - we should export it as a function, like we did with `sayHi` above. + +Now, let's consider a deeper example. Let's say, a module exports an object: @@ -153,60 +175,77 @@ import {admin} from './admin.js'; alert(admin.name); // Pete *!* -// Both 1.js and 2.js imported the same object +// Both 1.js and 2.js reference the same admin object // Changes made in 1.js are visible in 2.js */!* ``` -So, let's reiterate -- the module is executed only once. Exports are generated, and then they are shared between importers, so if something changes the `admin` object, other modules will see that . +As you can see, when `1.js` changes the `name` property in the imported `admin`, then `2.js` can see the new `admin.name`. + +That's exactly because the module is executed only once. Exports are generated, and then they are shared between importers, so if something changes the `admin` object, other importers will see that. -Such behavior is great for modules that require configuration. We can set required properties on the first import, and then in further imports it's ready. +**Such behavior is actually very convenient, because it allows us to *configure* modules.** -For instance, `admin.js` module may provide certain functionality, but expect the credentials to come into the `admin` object from outside: +In other words, a module can provide a generic functionality that needs a setup. E.g. authentication needs credentials. Then it can export a configuration object expecting the outer code to assign to it. + +Here's the classical pattern: +1. A module exports some means of configuration, e.g. a configuration object. +2. On the first import we initialize it, write to its properties. The top-level application script may do that. +3. Further imports use the module. + +For instance, the `admin.js` module may provide certain functionality (e.g. authentication), but expect the credentials to come into the `config` object from outside: ```js // 📁 admin.js -export let admin = { }; +export let config = { }; export function sayHi() { - alert(`Ready to serve, ${admin.name}!`); + alert(`Ready to serve, ${config.user}!`); } ``` -Now, in `init.js`, the first script of our app, we set `admin.name`. Then everyone will see it, including calls made from inside `admin.js` itself: +Here, `admin.js` exports the `config` object (initially empty, but may have default properties too). + +Then in `init.js`, the first script of our app, we import `config` from it and set `config.user`: ```js // 📁 init.js -import {admin} from './admin.js'; -admin.name = "Pete"; +import {config} from './admin.js'; +config.user = "Pete"; ``` -```js -// 📁 other.js -import {admin, sayHi} from './admin.js'; +...Now the module `admin.js` is configured. + +Further importers can call it, and it correctly shows the current user: -alert(admin.name); // *!*Pete*/!* +```js +// 📁 another.js +import {sayHi} from './admin.js'; sayHi(); // Ready to serve, *!*Pete*/!*! ``` + ### import.meta The object `import.meta` contains the information about the current module. -Its content depends on the environment. In the browser, it contains the url of the script, or a current webpage url if inside HTML: +Its content depends on the environment. In the browser, it contains the URL of the script, or a current webpage URL if inside HTML: ```html run height=0 <script type="module"> - alert(import.meta.url); // script url (url of the html page for an inline script) + alert(import.meta.url); // script URL + // for an inline script - the URL of the current HTML-page </script> ``` -### Top-level "this" is undefined +### In a module, "this" is undefined That's kind of a minor feature, but for completeness we should mention it. -In a module, top-level `this` is undefined, as opposed to a global object in non-module scripts: +In a module, top-level `this` is undefined. + +Compare it to non-module scripts, where `this` is a global object: ```html run height=0 <script> @@ -222,18 +261,18 @@ In a module, top-level `this` is undefined, as opposed to a global object in non There are also several browser-specific differences of scripts with `type="module"` compared to regular ones. -You may want skip those for now if you're reading for the first time, or if you don't use JavaScript in a browser. +You may want to skip this section for now if you're reading for the first time, or if you don't use JavaScript in a browser. ### Module scripts are deferred Module scripts are *always* deferred, same effect as `defer` attribute (described in the chapter [](info:script-async-defer)), for both external and inline scripts. In other words: -- external module scripts `<script type="module" src="...">` don't block HTML processing. -- module scripts wait until the HTML document is fully ready. -- relative order is maintained: scripts that go first in the document, execute first. +- downloading external module scripts `<script type="module" src="...">` doesn't block HTML processing, they load in parallel with other resources. +- module scripts wait until the HTML document is fully ready (even if they are tiny and load faster than HTML), and then run. +- relative order of scripts is maintained: scripts that go first in the document, execute first. -As a side-effect, module scripts always see HTML elements below them. +As a side effect, module scripts always "see" the fully loaded HTML-page, including HTML elements below them. For instance: @@ -245,9 +284,11 @@ For instance: // as modules are deferred, the script runs after the whole page is loaded </script> +Compare to regular script below: + <script> *!* - alert(typeof button); // Error: button is undefined, the script can't see elements below + alert(typeof button); // button is undefined, the script can't see elements below */!* // regular scripts run immediately, before the rest of the page is processed </script> @@ -255,19 +296,21 @@ For instance: <button id="button">Button</button> ``` -Please note: the second script actually works before the first! So we'll see `undefined` first, and then `object`. +Please note: the second script actually runs before the first! So we'll see `undefined` first, and then `object`. -That's because modules are deferred, so way wait for the document to be processed. The regular scripts runs immediately, so we saw its output first. +That's because modules are deferred, so we wait for the document to be processed. The regular script runs immediately, so we see its output first. -When using modules, we should be aware that HTML-document can show up before the JavaScript application is ready. Some functionality may not work yet. We should put transparent overlays or "loading indicators", or otherwise ensure that the visitor won't be confused because of it. +When using modules, we should be aware that the HTML page shows up as it loads, and JavaScript modules run after that, so the user may see the page before the JavaScript application is ready. Some functionality may not work yet. We should put "loading indicators", or otherwise ensure that the visitor won't be confused by that. ### Async works on inline scripts -Async attribute `<script async type="module">` is allowed on both inline and external scripts. Async scripts run immediately when imported modules are processed, independantly of other scripts or the HTML document. +For non-module scripts, the `async` attribute only works on external scripts. Async scripts run immediately when ready, independently of other scripts or the HTML document. + +For module scripts, it works on inline scripts as well. -For example, the script below has `async`, so it doesn't wait for anyone. +For example, the inline script below has `async`, so it doesn't wait for anything. -It performs the import (fetches `./analytics.js`) and runs when ready, even if HTML document is not finished yet, or if other scripts are still pending. +It performs the import (fetches `./analytics.js`) and runs when ready, even if the HTML document is not finished yet, or if other scripts are still pending. That's good for functionality that doesn't depend on anything, like counters, ads, document-level event listeners. @@ -283,16 +326,16 @@ That's good for functionality that doesn't depend on anything, like counters, ad ### External scripts -There are two notable differences of external module scripts: +External scripts that have `type="module"` are different in two aspects: -1. External scripts with same `src` run only once: +1. External scripts with the same `src` run only once: ```html <!-- the script my.js is fetched and executed only once --> <script type="module" src="my.js"></script> <script type="module" src="my.js"></script> ``` -2. External scripts that are fetched from another domain require [CORS](mdn:Web/HTTP/CORS) headers. In other words, if a module script is fetched from another domain, the remote server must supply a header `Access-Control-Allow-Origin: *` (may use fetching domain instead of `*`) to indicate that the fetch is allowed. +2. External scripts that are fetched from another origin (e.g. another site) require [CORS](mdn:Web/HTTP/CORS) headers, as described in the chapter <info:fetch-crossorigin>. In other words, if a module script is fetched from another origin, the remote server must supply a header `Access-Control-Allow-Origin` allowing the fetch. ```html <!-- another-site.com must supply Access-Control-Allow-Origin --> <!-- otherwise, the script won't execute --> @@ -301,21 +344,21 @@ There are two notable differences of external module scripts: That ensures better security by default. -### No bare modules allowed +### No "bare" modules allowed -In the browser, in scripts (not in HTML), `import` must get either a relative or absolute URL. So-called "bare" modules, without a path, are not allowed. +In the browser, `import` must get either a relative or absolute URL. Modules without any path are called "bare" modules. Such modules are not allowed in `import`. For instance, this `import` is invalid: ```js import {sayHi} from 'sayHi'; // Error, "bare" module -// must be './sayHi.js' or wherever the module is +// the module must have a path, e.g. './sayHi.js' or wherever the module is ``` -Certain environments, like Node.js or bundle tools allow bare modules, as they have own ways for finding modules and hooks to fine-tune them. But browsers do not support bare modules yet. +Certain environments, like Node.js or bundle tools allow bare modules, without any path, as they have their own ways for finding modules and hooks to fine-tune them. But browsers do not support bare modules yet. ### Compatibility, "nomodule" -Old browsers do not understand `type="module"`. Scripts of the unknown type are just ignored. For them, it's possible to provide a fallback using `nomodule` attribute: +Old browsers do not understand `type="module"`. Scripts of an unknown type are just ignored. For them, it's possible to provide a fallback using the `nomodule` attribute: ```html run <script type="module"> @@ -328,13 +371,6 @@ Old browsers do not understand `type="module"`. Scripts of the unknown type are </script> ``` -If we use bundle tools, then as modules are bundled together, their `import/export` statements are replaced by special bundler calls, so the resulting build does not require `type="module"`, and we can put it into a regular script: - -```html -<!-- Assuming we got bundle.js from a tool like Webpack --> -<script src="bundle.js"></script> -``` - ## Build tools In real-life, browser modules are rarely used in their "raw" form. Usually, we bundle them together with a special tool such as [Webpack](https://webpack.js.org/) and deploy to the production server. @@ -346,12 +382,19 @@ Build tools do the following: 1. Take a "main" module, the one intended to be put in `<script type="module">` in HTML. 2. Analyze its dependencies: imports and then imports of imports etc. 3. Build a single file with all modules (or multiple files, that's tunable), replacing native `import` calls with bundler functions, so that it works. "Special" module types like HTML/CSS modules are also supported. -4. In the process, other transforms and optimizations may be applied: +4. In the process, other transformations and optimizations may be applied: - Unreachable code removed. - Unused exports removed ("tree-shaking"). - Development-specific statements like `console` and `debugger` removed. - Modern, bleeding-edge JavaScript syntax may be transformed to older one with similar functionality using [Babel](https://babeljs.io/). - - The resulting file is minified (spaces removed, variables replaced with shorter named etc). + - The resulting file is minified (spaces removed, variables replaced with shorter names, etc). + +If we use bundle tools, then as scripts are bundled together into a single file (or few files), `import/export` statements inside those scripts are replaced by special bundler functions. So the resulting "bundled" script does not contain any `import/export`, it doesn't require `type="module"`, and we can put it into a regular script: + +```html +<!-- Assuming we got bundle.js from a tool like Webpack --> +<script src="bundle.js"></script> +``` That said, native modules are also usable. So we won't be using Webpack here: you can configure it later. @@ -359,16 +402,16 @@ That said, native modules are also usable. So we won't be using Webpack here: yo To summarize, the core concepts are: -1. A module is a file. To make `import/export` work, browsers need `<script type="module">`, that implies several differences: +1. A module is a file. To make `import/export` work, browsers need `<script type="module">`. Modules have several differences: - Deferred by default. - Async works on inline scripts. - - External scripts need CORS headers. + - To load external scripts from another origin (domain/protocol/port), CORS headers are needed. - Duplicate external scripts are ignored. 2. Modules have their own, local top-level scope and interchange functionality via `import/export`. 3. Modules always `use strict`. 4. Module code is executed only once. Exports are created once and shared between importers. -So, generally, when we use modules, each module implements the functionality and exports it. Then we use `import` to directly import it where it's needed. Browser loads and evaluates the scripts automatically. +When we use modules, each module implements the functionality and exports it. Then we use `import` to directly import it where it's needed. The browser loads and evaluates the scripts automatically. In production, people often use bundlers such as [Webpack](https://webpack.js.org) to bundle modules together for performance and other reasons. diff --git a/1-js/13-modules/02-import-export/article.md b/1-js/13-modules/02-import-export/article.md index d485a25b0..c383fa0a0 100644 --- a/1-js/13-modules/02-import-export/article.md +++ b/1-js/13-modules/02-import-export/article.md @@ -1,9 +1,14 @@ - # Export and Import +<<<<<<< HEAD Export og import direktiver er veldig allsidige. I forrige kapittel så vi en enklere bruk av disse, la oss nå utforske flere eksempler. +======= +Export and import directives have several syntax variants. + +In the previous article we saw a simple use, now let's explore more examples. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Export før deklareringer @@ -27,12 +32,20 @@ For eksempel, alle disse er gyldig bruk av eksportering: ``` ````smart header="No semicolons after export class/function" +<<<<<<< HEAD Vennligst legg merke til at `export` før en klasse, eller en funksjon vil ikke gjøre den til et [funksjons utrykk](info:function-expressions-arrows). Det er fremdeles en deklarasjon av en funksjon, bare eksportert. De fleste JavaScript stilguidene anbefaler bruk av semikolon etter utrykk, men ikke etter deklarering av funskjoner og klasser. That's why there should be no semicolons at the end of `export class` and `export function`. Dette er fordi det aldri skal være semikolon etter slutten av `export class` og `export function`. +======= +Please note that `export` before a class or a function does not make it a [function expression](info:function-expressions). It's still a function declaration, albeit exported. + +Most JavaScript style guides don't recommend semicolons after function and class declarations. + +That's why there's no need for a semicolon at the end of `export class` and `export function`: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js export function seyHi(user) { @@ -48,7 +61,7 @@ Vi kan også sette `export` separat fra et utrykk. Først deklarerer vi, også eksporterer vi: -```js +```js // 📁 say.js export function sayHi(user) { alert(`Hi, ${user}!`); @@ -67,7 +80,11 @@ export {sayHi, sayGoodbye}; // en liste av eksporterte variabler ## Import * +<<<<<<< HEAD Vanligvis, kan vi putte en liste av hva som skal importeres inni `import {...}`, slik som dette: +======= +Usually, we put a list of what to import in curly braces `import {...}`, like this: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js @@ -78,7 +95,11 @@ sayHi('John'); // Hi, John! sayGoodbye('John'); // Goodbye, John! ``` +<<<<<<< HEAD Men hvis denne listen er lang, kan vi importere alt som et objekt ved å bruke `import * as <obj>`, for eksempel: +======= +But if there's a lot to import, we can import everything as an object using `import * as <obj>`, for instance: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js @@ -94,6 +115,7 @@ Ved første øyekast, "importer alt" virker som en kul ting å gjøre, kort å s Vel, det er noen få grunner. +<<<<<<< HEAD 1. Moderne byggverktøy ([webpack](http://webpack.github.io) og andre) pakker moduler sammen og optimaliserer de til å kunne raske laste og fjeren unyttige ting. La oss si, at vi la til et 3. partibibliotek `lib.js` inn i prosjektet med mange funksjoner: @@ -118,6 +140,22 @@ Vel, det er noen få grunner. Vi kan også bruke `as` til å importere under forskjellige alias. For eksempel, la oss importere `siHei` inn i den lokale variabelen `hei` for korthet i koden, og det samme for `siFarvel`: +======= +1. Explicitly listing what to import gives shorter names: `sayHi()` instead of `say.sayHi()`. +2. Explicit list of imports gives better overview of the code structure: what is used and where. It makes code support and refactoring easier. + +```smart header="Don't be afraid to import too much" +Modern build tools, such as [webpack](https://webpack.js.org/) and others, bundle modules together and optimize them to speedup loading. They also remove unused imports. + +For instance, if you `import * as library` from a huge code library, and then use only few methods, then unused ones [will not be included](https://github.com/webpack/webpack/tree/main/examples/harmony-unused#examplejs) into the optimized bundle. +``` + +## Import "as" + +We can also use `as` to import under different names. + +For instance, let's import `sayHi` into the local variable `hi` for brevity, and import `sayBye` as `bye`: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js @@ -141,26 +179,43 @@ La oss eksportere funksjoner som `hei` og `farvel`: export {sayHi as hi, sayGoodbye as goodbye}; ``` +<<<<<<< HEAD Nå er `hi` og `goodbye` offisielle navn for parter utenfor: +======= +Now `hi` and `bye` are official names for outsiders, to be used in imports: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js import * as say from './say.js'; +<<<<<<< HEAD say.hi('John'); // Hi, John! say.goodbye('John'); // Goodbye, John! +======= +say.*!*hi*/!*('John'); // Hello, John! +say.*!*bye*/!*('John'); // Bye, John! +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ``` -## export default +## Export default +<<<<<<< HEAD Så langt har vi sett på hvordan man importerer/eksporterer flere ting, valgfritt som (`as`) andre navn. I praksis, modulere inneholder enten: - Et bibliotek, samling av funksjoner, som `lib.js`. - Eller et objekt som `class User` er beskrevet i `user.js`, hele modulen har kun denne klassen. +======= +In practice, there are mainly two kinds of modules. + +1. Modules that contain a library, pack of functions, like `say.js` above. +2. Modules that declare a single entity, e.g. a module `user.js` exports only `class User`. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Som oftest, er den andre måten å ta ibruk foretrukket, sånn at hver "ting" oppholdes i sin egen modul. +<<<<<<< HEAD Naturligvis, dette krever en hel del filer, som alle vil ha sin egen modul, men dette er ikke et problem. Faktisk, så blir navigering av kodebasen lettere, hvis filene er gitt bra navn og strukturert godt i egne mapper. Moduler gir spesielle `export default` syntaks for å gjøre at "en ting per modul" ser bedre ut. @@ -171,6 +226,13 @@ Det krever følgende `export` og `import` uttrykk: 2. Kall `import` uten krøllparenteser. For eksempel, her `bruker.js` exports `class Bruker`: +======= +Naturally, that requires a lot of files, as everything wants its own module, but that's not a problem at all. Actually, code navigation becomes easier if files are well-named and structured into folders. + +Modules provide a special `export default` ("the default export") syntax to make the "one thing per module" way look better. + +Put `export default` before the entity to export: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 user.js @@ -181,7 +243,13 @@ export *!*default*/!* class User { // kun tilføy "default" } ``` +<<<<<<< HEAD ...Og `main.js` importerer det: +======= +There may be only one `export default` per file. + +...And then import it without curly braces: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js @@ -189,18 +257,28 @@ import *!*User*/!* from './user.js'; // ikke {User}, kun User new User('John'); ``` +<<<<<<< HEAD Importsetninger uten krøllparentes ser finere ut. En vanlig feil når man begynner å bruke moduler er å glemme krøllparentes helt. Så, husk, `import` trenger krøllparentes for navngitte importsetninger og trenger dem ikke ikke default import. +======= +Imports without curly braces look nicer. A common mistake when starting to use modules is to forget curly braces at all. So, remember, `import` needs curly braces for named exports and doesn't need them for the default one. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b | Navngitt export | Default export | |--------------|----------------| | `export class User {...}` | `export default class User {...}` | | `import { User } from ...` | `import User from ...`| +<<<<<<< HEAD Naturligvis kan det kun være en "default" export per fil. Vi kan ha både default og navngitt eksporteringer i en enkelt modul, men i praksis vil folk blande disse to. En modul har enten navngitte eksporteringer eller en default. **En annen ting å bite seg merke i er at navngitt eksporteringer må (naturligvis) ha et navn, mens `export default` kan være anonym.** +======= +Technically, we may have both default and named exports in a single module, but in practice people usually don't mix them. A module has either named exports or the default one. + +As there may be at most one default export per file, the exported entity may have no name. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b For eksempel, disse er alle helt gyldige eksporteringer: @@ -208,17 +286,36 @@ For eksempel, disse er alle helt gyldige eksporteringer: export default class { // ingen klassenavn constructor() { ... } } +``` +<<<<<<< HEAD export default function(user) { // ingen funksjonsnavn alert(`Hi, ${user}!`); +======= +```js +export default function(user) { // no function name + alert(`Hello, ${user}!`); +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b } +``` +<<<<<<< HEAD // eksporter en enkelt verdi, uten å definere en variabel export default ['Jan', 'Feb', 'Mar','Apr', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; ``` Det går helt fint, fordi `export default` finnes det kun en av per fil, så `import` vet alltid hva som skal importeres. I motsetning til det, og utelater et navn for navngitte importeringer ville vært en feil i koden: +======= +```js +// export a single value, without making a variable +export default ['Jan', 'Feb', 'Mar','Apr', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; +``` + +Not giving a name is fine, because there is only one `export default` per file, so `import` without curly braces knows what to import. + +Without `default`, such an export would give an error: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js export class { // Feil! (ikke-default eksportering trenger et navn) @@ -226,22 +323,36 @@ export class { // Feil! (ikke-default eksportering trenger et navn) } ``` -### "Default" alias +### The "default" name +<<<<<<< HEAD Stikkordet "default" er et slags alias for default export, for når vi trenger å angi en referanse for den. For example, if we already have a function declared, that's how to `export default` it: For eksempel, når vi allerede har deklarert en funksjon, er dette hvordan `export default` den: +======= +In some situations the `default` keyword is used to reference the default export. + +For example, to export a function separately from its definition: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js function sayHi(user) { alert(`Hi, ${user}!`); } +<<<<<<< HEAD export {sayHi as default}; // samme som hvis vi hadde lagt til "export default" før funksjonen ``` Eller, la oss si at modulen `user.js` eksporterer en hoved "default" ting og et par navngitte noen (sjeldent tilfelle, men skjer av og til): +======= +// same as if we added "export default" before the function +export {sayHi as default}; +``` + +Or, another situation, let's say a module `user.js` exports one main "default" thing, and a few named ones (rarely the case, but it happens): +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 user.js @@ -265,16 +376,21 @@ import { *!*defualt as User*/!*, sayHi } from './user.js'; new User('John'); ``` +<<<<<<< HEAD Eller hvis vi vurderer å importere `*` som et objekt, da må `default` egenskapen være eksagt default export: +======= +And, finally, if importing everything `*` as an object, then the `default` property is exactly the default export: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 main.js import * as User from './user.js'; -let User = user.default; +let User = user.default; // the default export new User('John'); ``` +<<<<<<< HEAD ### Burde jeg bruker default exports? Vi må være forsiktige når vi bruker default exports, fordi disse kan være til en viss grad veldig annerledes å vedlikeholde. @@ -295,6 +411,30 @@ import MyUser from './user.js'; // dette kunne vært importer hvasomhelst... , o Slik at det blir så lite frihet som mulig til at dette kan brukes feil, slik at team medlemmer kan bruke forskjellige navn for den samme tingen. Vanligvis, for å unngå dette og holde koden mer konsistent, er det regel for at importerte variabler burde stemmer overens med sitt filnavn, f.eks: +======= +### A word against default exports + +Named exports are explicit. They exactly name what they import, so we have that information from them; that's a good thing. + +Named exports force us to use exactly the right name to import: + +```js +import {User} from './user.js'; +// import {MyUser} won't work, the name must be {User} +``` + +...While for a default export, we always choose the name when importing: + +```js +import User from './user.js'; // works +import MyUser from './user.js'; // works too +// could be import Anything... and it'll still work +``` + +So team members may use different names to import the same thing, and that's not good. + +Usually, to avoid that and keep the code consistent, there's a rule that imported variables should correspond to file names, e.g: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js import User from './user.js'; @@ -303,7 +443,11 @@ import func from '/path/to/func.js'; ... ``` +<<<<<<< HEAD En annen løsning ville vært å bruke navngitte exports overalt. Til og med hvis kun en enkelt fil er eksportert, er den fremdeles eksportert under et navn, uten `default`. +======= +Still, some teams consider it a serious drawback of default exports. So they prefer to always use named exports. Even if only a single thing is exported, it's still exported under a name, without `default`. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Dette gjør også re-export (se under) litt enklere å forstå. @@ -313,6 +457,7 @@ Dette gjør også re-export (se under) litt enklere å forstå. ```js +<<<<<<< HEAD export { sayHi } from './say.js'; export { default as User } from './user.js'; ``` @@ -322,72 +467,136 @@ Hva er poenget, hvorfor trenger vi dette? La oss se på et praktisk bruksområde Se for deg, vi skriver en "pakke": en mappe med mange moduler, mest til bruk internt, med noe funksjonalitet eksportert utad (verktøy som NPM lar oss publisere og distribuere pakker, men det har ikke noe å si her). En mappestruktur kan være slik: +======= +export {sayHi} from './say.js'; // re-export sayHi + +export {default as User} from './user.js'; // re-export default +``` + +Why would that be needed? Let's see a practical use case. + +Imagine, we're writing a "package": a folder with a lot of modules, with some of the functionality exported outside (tools like NPM allow us to publish and distribute such packages, but we don't have to use them), and many modules are just "helpers", for internal use in other package modules. + +The file structure could be like this: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ``` auth/ - index.js - user.js - helpers.js - tests/ - login.js - providers/ - github.js - facebook.js - ... + index.js + user.js + helpers.js + tests/ + login.js + providers/ + github.js + facebook.js + ... ``` +<<<<<<< HEAD Vi vil eksponere pakkens funksjonalitet via et enkelt entry point, "hovedfilen" `auth/inde.js`, kan være slik: +======= +We'd like to expose the package functionality via a single entry point. + +In other words, a person who would like to use our package, should import only from the "main file" `auth/index.js`. + +Like this: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js import {login, logout} from 'auth/index.js' ``` +<<<<<<< HEAD Ideen er at parter fra utsiden, utviklere som bruker pakken vår, ikke bør tenke på dens interne struktur. De skal ikke måtte lete etter filer inne i vår pakke. Vi eksportert kun det som er nødvendig gjennom `auth/inde.js` og holder resten skjult fra nysgjerrige øyne Nå som den faktiske eksporterte funksjonaliteten er spredt blant pakken, kan vi samle og "re-export" den i `auth/index.js`: +======= +The "main file", `auth/index.js` exports all the functionality that we'd like to provide in our package. + +The idea is that outsiders, other programmers who use our package, should not meddle with its internal structure, search for files inside our package folder. We export only what's necessary in `auth/index.js` and keep the rest hidden from prying eyes. + +As the actual exported functionality is scattered among the package, we can import it into `auth/index.js` and export from it: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 auth/index.js + +// import login/logout and immediately export them import {login, logout} from './helpers.js'; export {login, logout}; +// import default as User and export it import User from './user.js'; export {User}; - -import Github from './providers/github.js'; -export {Github}; ... ``` +<<<<<<< HEAD "Re-eksportering" er bare en måte å gjøre slik på: +======= +Now users of our package can `import {login} from "auth/index.js"`. + +The syntax `export ... from ...` is just a shorter notation for such import-export: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js // 📁 auth/index.js +// re-export login/logout export {login, logout} from './helpers.js'; -// or, to re-export all helpers, we could use: -// export * from './helpers.js'; +// re-export the default export as User export {default as User} from './user.js'; - -export {default as Github} from './providers/github.js'; ... ``` +<<<<<<< HEAD ````warn header="Re-eksporering default kan være vrient" Vær grei å merk: `export User from './user.js'` vill ikke virke. Dette er faktisk en syntaks feil. For å re-export en default export, må vi referensere den eksplisitt slik som `{default as ...}`, slik som eksempelet over. I tillegg, er det en annen særhet: `export * from './user.js'` re-exports er kun navngitte eksporteringer, ekskludert standard eksport. Men igjen, vi må nevne den eksplisitt. For eksempel, for å re-export alt, trenger vi to uttrykk: +======= +The notable difference of `export ... from` compared to `import/export` is that re-exported modules aren't available in the current file. So inside the above example of `auth/index.js` we can't use re-exported `login/logout` functions. + +### Re-exporting the default export + +The default export needs separate handling when re-exporting. + +Let's say we have `user.js` with the `export default class User` and would like to re-export it: + +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```js -export * from './module.js'; // to re-export named exports -export {default} from './module.js'; // to re-export default +// 📁 user.js +export default class User { + // ... +} ``` +<<<<<<< HEAD Standarden burde være nevnt eksplisitt kun når man re-eksporterer: `import * as obj` virker fint. Dette importerer en standard export som `obj.default`. Slik at det er en liten asymmetry mellom import og export konstruksjoner her. ```` +======= +We can come across two problems with it: + +1. `export User from './user.js'` won't work. That would lead to a syntax error. + + To re-export the default export, we have to write `export {default as User}`, as in the example above. + +2. `export * from './user.js'` re-exports only named exports, but ignores the default one. + + If we'd like to re-export both named and default exports, then two statements are needed: + ```js + export * from './user.js'; // to re-export named exports + export {default} from './user.js'; // to re-export the default export + ``` + +Such oddities of re-exporting a default export are one of the reasons why some developers don't like default exports and prefer named ones. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ## Sammendrag +<<<<<<< HEAD Det er følgende typer av `export`: - Før deklarering: @@ -410,9 +619,36 @@ Import: - `import * as obj from "mod"` - Kun hent/evaluer modulen, ikke importer: - `import "mod"` +======= +Here are all types of `export` that we covered in this and previous articles. + +You can check yourself by reading them and recalling what they mean: + +- Before declaration of a class/function/..: + - `export [default] class/function/variable ...` +- Standalone export: + - `export {x [as y], ...}`. +- Re-export: + - `export {x [as y], ...} from "module"` + - `export * from "module"` (doesn't re-export default). + - `export {default [as y]} from "module"` (re-export default). + +Import: -We can put import/export statements below or after other code, that doesn't matter. +- Importing named exports: + - `import {x [as y], ...} from "module"` +- Importing the default export: + - `import x from "module"` + - `import {default as x} from "module"` +- Import all: + - `import * as obj from "module"` +- Import the module (its code runs), but do not assign any of its exports to variables: + - `import "module"` +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b +We can put `import/export` statements at the top or at the bottom of a script, that doesn't matter. + +<<<<<<< HEAD Så dette er teknisk rett: ```js sayHi(); @@ -421,6 +657,18 @@ import {sayHi} from './say.js'; // import på slutten av filen ``` I praksis så er imports vanligvis deklarert i begynnelsen av hver enkelt filt, men det er kun av praktiske årsaker. +======= +So, technically this code is fine: +```js +sayHi(); + +// ... + +import {sayHi} from './say.js'; // import at the end of the file +``` + +In practice imports are usually at the start of the file, but that's only for more convenience. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b **Vær grei å legg merke til at import/export uttrykk ikke vil hvis de er plasser inne i `{...}`.** @@ -433,4 +681,8 @@ if (something) { ...Men hva hvis vi virkelig trenger å importere noe basert på en betingelse? Eller på riktig tidspunkt i koden? Slik som, laste inn en modul på forespørsel, når den virkelig trengs? +<<<<<<< HEAD Vi skal se på dynamiske import i neste kapittel. +======= +We'll see dynamic imports in the next article. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/1-js/13-modules/03-modules-dynamic-imports/article.md b/1-js/13-modules/03-modules-dynamic-imports/article.md index 5d2538729..e48144a3e 100644 --- a/1-js/13-modules/03-modules-dynamic-imports/article.md +++ b/1-js/13-modules/03-modules-dynamic-imports/article.md @@ -1,9 +1,6 @@ - # Dynamic imports -Export and import statements that we covered in previous chaters are called "static". - -That's because they are indeed static. The syntax is very strict. +Export and import statements that we covered in previous chapters are called "static". The syntax is very simple and strict. First, we can't dynamically generate any parameters of `import`. @@ -25,30 +22,77 @@ if(...) { } ``` -That's because, import/export aim to provide a backbone for the code structure. That's a good thing, as code structure can be analyzed, modules can be gathered and bundled together, unused exports can be removed (tree-shaken). That's possible only because everything is fixed. +That's because `import`/`export` aim to provide a backbone for the code structure. That's a good thing, as code structure can be analyzed, modules can be gathered and bundled into one file by special tools, unused exports can be removed ("tree-shaken"). That's possible only because the structure of imports/exports is simple and fixed. -But how do we import a module dynamically, on-demand? +But how can we import a module dynamically, on-demand? -## The import() function +## The import() expression -The `import(module)` function can be called from anywhere. It returns a promise that resolves into a module object. +The `import(module)` expression loads the module and returns a promise that resolves into a module object that contains all its exports. It can be called from any place in the code. -The usage pattern looks like this: +We can use it dynamically in any place of the code, for instance: -```js run -let modulePath = prompt("Module path?"); +```js +let modulePath = prompt("Which module to load?"); import(modulePath) .then(obj => <module object>) - .catch(err => <loading error, no such module?>) + .catch(err => <loading error, e.g. if no such module>) ``` Or, we could use `let module = await import(modulePath)` if inside an async function. -Like this: +For instance, if we have the following module `say.js`: + +```js +// 📁 say.js +export function hi() { + alert(`Hello`); +} + +export function bye() { + alert(`Bye`); +} +``` + +...Then dynamic import can be like this: + +```js +let {hi, bye} = await import('./say.js'); + +hi(); +bye(); +``` + +Or, if `say.js` has the default export: + +```js +// 📁 say.js +export default function() { + alert("Module loaded (export default)!"); +} +``` + +...Then, in order to access it, we can use `default` property of the module object: + +```js +let obj = await import('./say.js'); +let say = obj.default; +// or, in one line: let {default: say} = await import('./say.js'); + +say(); +``` + +Here's the full example: [codetabs src="say" current="index.html"] -So, dynamic imports are very simple to use. +```smart +Dynamic imports work in regular scripts, they don't require `script type="module"`. +``` + +```smart +Although `import()` looks like a function call, it's a special syntax that just happens to use parentheses (similar to `super()`). -Also, dynamic imports work in regular scripts, they don't require `script type="module"`. +So we can't copy `import` to a variable or use `call/apply` with it. It's not a function. +``` diff --git a/1-js/99-js-misc/01-proxy/01-error-nonexisting/solution.md b/1-js/99-js-misc/01-proxy/01-error-nonexisting/solution.md new file mode 100644 index 000000000..9db69cb2f --- /dev/null +++ b/1-js/99-js-misc/01-proxy/01-error-nonexisting/solution.md @@ -0,0 +1,23 @@ + +```js run +let user = { + name: "John" +}; + +function wrap(target) { + return new Proxy(target, { + get(target, prop, receiver) { + if (prop in target) { + return Reflect.get(target, prop, receiver); + } else { + throw new ReferenceError(`Property doesn't exist: "${prop}"`) + } + } + }); +} + +user = wrap(user); + +alert(user.name); // John +alert(user.age); // ReferenceError: Property doesn't exist: "age" +``` diff --git a/1-js/99-js-misc/01-proxy/01-error-nonexisting/task.md b/1-js/99-js-misc/01-proxy/01-error-nonexisting/task.md new file mode 100644 index 000000000..47985e1a7 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/01-error-nonexisting/task.md @@ -0,0 +1,32 @@ +# Error on reading non-existent property + +Usually, an attempt to read a non-existent property returns `undefined`. + +Create a proxy that throws an error for an attempt to read of a non-existent property instead. + +That can help to detect programming mistakes early. + +Write a function `wrap(target)` that takes an object `target` and return a proxy that adds this functionality aspect. + +That's how it should work: + +```js +let user = { + name: "John" +}; + +function wrap(target) { + return new Proxy(target, { +*!* + /* your code */ +*/!* + }); +} + +user = wrap(user); + +alert(user.name); // John +*!* +alert(user.age); // ReferenceError: Property doesn't exist: "age" +*/!* +``` diff --git a/1-js/99-js-misc/01-proxy/02-array-negative/solution.md b/1-js/99-js-misc/01-proxy/02-array-negative/solution.md new file mode 100644 index 000000000..207205501 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/02-array-negative/solution.md @@ -0,0 +1,19 @@ + +```js run +let array = [1, 2, 3]; + +array = new Proxy(array, { + get(target, prop, receiver) { + if (prop < 0) { + // even if we access it like arr[1] + // prop is a string, so need to convert it to number + prop = +prop + target.length; + } + return Reflect.get(target, prop, receiver); + } +}); + + +alert(array[-1]); // 3 +alert(array[-2]); // 2 +``` diff --git a/1-js/99-js-misc/01-proxy/02-array-negative/task.md b/1-js/99-js-misc/01-proxy/02-array-negative/task.md new file mode 100644 index 000000000..9b0b13f58 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/02-array-negative/task.md @@ -0,0 +1,33 @@ + +# Accessing array[-1] + +In some programming languages, we can access array elements using negative indexes, counted from the end. + +Like this: + +```js +let array = [1, 2, 3]; + +array[-1]; // 3, the last element +array[-2]; // 2, one step from the end +array[-3]; // 1, two steps from the end +``` + +In other words, `array[-N]` is the same as `array[array.length - N]`. + +Create a proxy to implement that behavior. + +That's how it should work: + +```js +let array = [1, 2, 3]; + +array = new Proxy(array, { + /* your code */ +}); + +alert( array[-1] ); // 3 +alert( array[-2] ); // 2 + +// Other array functionality should be kept "as is" +``` diff --git a/1-js/99-js-misc/01-proxy/03-observable/solution.md b/1-js/99-js-misc/01-proxy/03-observable/solution.md new file mode 100644 index 000000000..c0797a856 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/03-observable/solution.md @@ -0,0 +1,40 @@ +The solution consists of two parts: + +1. Whenever `.observe(handler)` is called, we need to remember the handler somewhere, to be able to call it later. We can store handlers right in the object, using our symbol as the property key. +2. We need a proxy with `set` trap to call handlers in case of any change. + +```js run +let handlers = Symbol('handlers'); + +function makeObservable(target) { + // 1. Initialize handlers store + target[handlers] = []; + + // Store the handler function in array for future calls + target.observe = function(handler) { + this[handlers].push(handler); + }; + + // 2. Create a proxy to handle changes + return new Proxy(target, { + set(target, property, value, receiver) { + let success = Reflect.set(...arguments); // forward the operation to object + if (success) { // if there were no error while setting the property + // call all handlers + target[handlers].forEach(handler => handler(property, value)); + } + return success; + } + }); +} + +let user = {}; + +user = makeObservable(user); + +user.observe((key, value) => { + alert(`SET ${key}=${value}`); +}); + +user.name = "John"; +``` diff --git a/1-js/99-js-misc/01-proxy/03-observable/task.md b/1-js/99-js-misc/01-proxy/03-observable/task.md new file mode 100644 index 000000000..754d9f3bd --- /dev/null +++ b/1-js/99-js-misc/01-proxy/03-observable/task.md @@ -0,0 +1,27 @@ + +# Observable + +Create a function `makeObservable(target)` that "makes the object observable" by returning a proxy. + +Here's how it should work: + +```js run +function makeObservable(target) { + /* your code */ +} + +let user = {}; +user = makeObservable(user); + +user.observe((key, value) => { + alert(`SET ${key}=${value}`); +}); + +user.name = "John"; // alerts: SET name=John +``` + +In other words, an object returned by `makeObservable` is just like the original one, but also has the method `observe(handler)` that sets `handler` function to be called on any property change. + +Whenever a property changes, `handler(key, value)` is called with the name and value of the property. + +P.S. In this task, please only take care about writing to a property. Other operations can be implemented in a similar way. diff --git a/1-js/99-js-misc/01-proxy/article.md b/1-js/99-js-misc/01-proxy/article.md new file mode 100644 index 000000000..1f84912e5 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/article.md @@ -0,0 +1,1034 @@ +# Proxy and Reflect + +A `Proxy` object wraps another object and intercepts operations, like reading/writing properties and others, optionally handling them on its own, or transparently allowing the object to handle them. + +Proxies are used in many libraries and some browser frameworks. We'll see many practical applications in this article. + +## Proxy + +The syntax: + +```js +let proxy = new Proxy(target, handler) +``` + +- `target` -- is an object to wrap, can be anything, including functions. +- `handler` -- proxy configuration: an object with "traps", methods that intercept operations. - e.g. `get` trap for reading a property of `target`, `set` trap for writing a property into `target`, and so on. + +For operations on `proxy`, if there's a corresponding trap in `handler`, then it runs, and the proxy has a chance to handle it, otherwise the operation is performed on `target`. + +As a starting example, let's create a proxy without any traps: + +```js run +let target = {}; +let proxy = new Proxy(target, {}); // empty handler + +proxy.test = 5; // writing to proxy (1) +alert(target.test); // 5, the property appeared in target! + +alert(proxy.test); // 5, we can read it from proxy too (2) + +for(let key in proxy) alert(key); // test, iteration works (3) +``` + +As there are no traps, all operations on `proxy` are forwarded to `target`. + +1. A writing operation `proxy.test=` sets the value on `target`. +2. A reading operation `proxy.test` returns the value from `target`. +3. Iteration over `proxy` returns values from `target`. + +As we can see, without any traps, `proxy` is a transparent wrapper around `target`. + + + +`Proxy` is a special "exotic object". It doesn't have own properties. With an empty `handler` it transparently forwards operations to `target`. + +To activate more capabilities, let's add traps. + +What can we intercept with them? + +For most operations on objects, there's a so-called "internal method" in the JavaScript specification that describes how it works at the lowest level. For instance `[[Get]]`, the internal method to read a property, `[[Set]]`, the internal method to write a property, and so on. These methods are only used in the specification, we can't call them directly by name. + +Proxy traps intercept invocations of these methods. They are listed in the [Proxy specification](https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots) and in the table below. + +For every internal method, there's a trap in this table: the name of the method that we can add to the `handler` parameter of `new Proxy` to intercept the operation: + +| Internal Method | Handler Method | Triggers when... | +|-----------------|----------------|-------------| +| `[[Get]]` | `get` | reading a property | +| `[[Set]]` | `set` | writing to a property | +| `[[HasProperty]]` | `has` | `in` operator | +| `[[Delete]]` | `deleteProperty` | `delete` operator | +| `[[Call]]` | `apply` | function call | +| `[[Construct]]` | `construct` | `new` operator | +| `[[GetPrototypeOf]]` | `getPrototypeOf` | [Object.getPrototypeOf](mdn:/JavaScript/Reference/Global_Objects/Object/getPrototypeOf) | +| `[[SetPrototypeOf]]` | `setPrototypeOf` | [Object.setPrototypeOf](mdn:/JavaScript/Reference/Global_Objects/Object/setPrototypeOf) | +| `[[IsExtensible]]` | `isExtensible` | [Object.isExtensible](mdn:/JavaScript/Reference/Global_Objects/Object/isExtensible) | +| `[[PreventExtensions]]` | `preventExtensions` | [Object.preventExtensions](mdn:/JavaScript/Reference/Global_Objects/Object/preventExtensions) | +| `[[DefineOwnProperty]]` | `defineProperty` | [Object.defineProperty](mdn:/JavaScript/Reference/Global_Objects/Object/defineProperty), [Object.defineProperties](mdn:/JavaScript/Reference/Global_Objects/Object/defineProperties) | +| `[[GetOwnProperty]]` | `getOwnPropertyDescriptor` | [Object.getOwnPropertyDescriptor](mdn:/JavaScript/Reference/Global_Objects/Object/getOwnPropertyDescriptor), `for..in`, `Object.keys/values/entries` | +| `[[OwnPropertyKeys]]` | `ownKeys` | [Object.getOwnPropertyNames](mdn:/JavaScript/Reference/Global_Objects/Object/getOwnPropertyNames), [Object.getOwnPropertySymbols](mdn:/JavaScript/Reference/Global_Objects/Object/getOwnPropertySymbols), `for..in`, `Object.keys/values/entries` | + +```warn header="Invariants" +JavaScript enforces some invariants -- conditions that must be fulfilled by internal methods and traps. + +Most of them are for return values: +- `[[Set]]` must return `true` if the value was written successfully, otherwise `false`. +- `[[Delete]]` must return `true` if the value was deleted successfully, otherwise `false`. +- ...and so on, we'll see more in examples below. + +There are some other invariants, like: +- `[[GetPrototypeOf]]`, applied to the proxy object must return the same value as `[[GetPrototypeOf]]` applied to the proxy object's target object. In other words, reading prototype of a proxy must always return the prototype of the target object. + +Traps can intercept these operations, but they must follow these rules. + +Invariants ensure correct and consistent behavior of language features. The full invariants list is in [the specification](https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots). You probably won't violate them if you're not doing something weird. +``` + +Let's see how that works in practical examples. + +## Default value with "get" trap + +The most common traps are for reading/writing properties. + +To intercept reading, the `handler` should have a method `get(target, property, receiver)`. + +It triggers when a property is read, with following arguments: + +- `target` -- is the target object, the one passed as the first argument to `new Proxy`, +- `property` -- property name, +- `receiver` -- if the target property is a getter, then `receiver` is the object that's going to be used as `this` in its call. Usually that's the `proxy` object itself (or an object that inherits from it, if we inherit from proxy). Right now we don't need this argument, so it will be explained in more detail later. + +Let's use `get` to implement default values for an object. + +We'll make a numeric array that returns `0` for nonexistent values. + +Usually when one tries to get a non-existing array item, they get `undefined`, but we'll wrap a regular array into the proxy that traps reading and returns `0` if there's no such property: + +```js run +let numbers = [0, 1, 2]; + +numbers = new Proxy(numbers, { + get(target, prop) { + if (prop in target) { + return target[prop]; + } else { + return 0; // default value + } + } +}); + +*!* +alert( numbers[1] ); // 1 +alert( numbers[123] ); // 0 (no such item) +*/!* +``` + +As we can see, it's quite easy to do with a `get` trap. + +We can use `Proxy` to implement any logic for "default" values. + +Imagine we have a dictionary, with phrases and their translations: + +```js run +let dictionary = { + 'Hello': 'Hola', + 'Bye': 'Adiós' +}; + +alert( dictionary['Hello'] ); // Hola +alert( dictionary['Welcome'] ); // undefined +``` + +Right now, if there's no phrase, reading from `dictionary` returns `undefined`. But in practice, leaving a phrase untranslated is usually better than `undefined`. So let's make it return an untranslated phrase in that case instead of `undefined`. + +To achieve that, we'll wrap `dictionary` in a proxy that intercepts reading operations: + +```js run +let dictionary = { + 'Hello': 'Hola', + 'Bye': 'Adiós' +}; + +dictionary = new Proxy(dictionary, { +*!* + get(target, phrase) { // intercept reading a property from dictionary +*/!* + if (phrase in target) { // if we have it in the dictionary + return target[phrase]; // return the translation + } else { + // otherwise, return the non-translated phrase + return phrase; + } + } +}); + +// Look up arbitrary phrases in the dictionary! +// At worst, they're not translated. +alert( dictionary['Hello'] ); // Hola +*!* +alert( dictionary['Welcome to Proxy']); // Welcome to Proxy (no translation) +*/!* +``` + +````smart +Please note how the proxy overwrites the variable: + +```js +dictionary = new Proxy(dictionary, ...); +``` + +The proxy should totally replace the target object everywhere. No one should ever reference the target object after it got proxied. Otherwise it's easy to mess up. +```` + +## Validation with "set" trap + +Let's say we want an array exclusively for numbers. If a value of another type is added, there should be an error. + +The `set` trap triggers when a property is written. + +`set(target, property, value, receiver)`: + +- `target` -- is the target object, the one passed as the first argument to `new Proxy`, +- `property` -- property name, +- `value` -- property value, +- `receiver` -- similar to `get` trap, matters only for setter properties. + +The `set` trap should return `true` if setting is successful, and `false` otherwise (triggers `TypeError`). + +Let's use it to validate new values: + +```js run +let numbers = []; + +numbers = new Proxy(numbers, { // (*) +*!* + set(target, prop, val) { // to intercept property writing +*/!* + if (typeof val == 'number') { + target[prop] = val; + return true; + } else { + return false; + } + } +}); + +numbers.push(1); // added successfully +numbers.push(2); // added successfully +alert("Length is: " + numbers.length); // 2 + +*!* +numbers.push("test"); // TypeError ('set' on proxy returned false) +*/!* + +alert("This line is never reached (error in the line above)"); +``` + +Please note: the built-in functionality of arrays is still working! Values are added by `push`. The `length` property auto-increases when values are added. Our proxy doesn't break anything. + +We don't have to override value-adding array methods like `push` and `unshift`, and so on, to add checks in there, because internally they use the `[[Set]]` operation that's intercepted by the proxy. + +So the code is clean and concise. + +```warn header="Don't forget to return `true`" +As said above, there are invariants to be held. + +For `set`, it must return `true` for a successful write. + +If we forget to do it or return any falsy value, the operation triggers `TypeError`. +``` + +## Iteration with "ownKeys" and "getOwnPropertyDescriptor" + +`Object.keys`, `for..in` loop and most other methods that iterate over object properties use `[[OwnPropertyKeys]]` internal method (intercepted by `ownKeys` trap) to get a list of properties. + +Such methods differ in details: +- `Object.getOwnPropertyNames(obj)` returns non-symbol keys. +- `Object.getOwnPropertySymbols(obj)` returns symbol keys. +- `Object.keys/values()` returns non-symbol keys/values with `enumerable` flag (property flags were explained in the article <info:property-descriptors>). +- `for..in` loops over non-symbol keys with `enumerable` flag, and also prototype keys. + +...But all of them start with that list. + +In the example below we use `ownKeys` trap to make `for..in` loop over `user`, and also `Object.keys` and `Object.values`, to skip properties starting with an underscore `_`: + +```js run +let user = { + name: "John", + age: 30, + _password: "***" +}; + +user = new Proxy(user, { +*!* + ownKeys(target) { +*/!* + return Object.keys(target).filter(key => !key.startsWith('_')); + } +}); + +// "ownKeys" filters out _password +for(let key in user) alert(key); // name, then: age + +// same effect on these methods: +alert( Object.keys(user) ); // name,age +alert( Object.values(user) ); // John,30 +``` + +So far, it works. + +Although, if we return a key that doesn't exist in the object, `Object.keys` won't list it: + +```js run +let user = { }; + +user = new Proxy(user, { +*!* + ownKeys(target) { +*/!* + return ['a', 'b', 'c']; + } +}); + +alert( Object.keys(user) ); // <empty> +``` + +Why? The reason is simple: `Object.keys` returns only properties with the `enumerable` flag. To check for it, it calls the internal method `[[GetOwnProperty]]` for every property to get [its descriptor](info:property-descriptors). And here, as there's no property, its descriptor is empty, no `enumerable` flag, so it's skipped. + +For `Object.keys` to return a property, we need it to either exist in the object, with the `enumerable` flag, or we can intercept calls to `[[GetOwnProperty]]` (the trap `getOwnPropertyDescriptor` does it), and return a descriptor with `enumerable: true`. + +Here's an example of that: + +```js run +let user = { }; + +user = new Proxy(user, { + ownKeys(target) { // called once to get a list of properties + return ['a', 'b', 'c']; + }, + + getOwnPropertyDescriptor(target, prop) { // called for every property + return { + enumerable: true, + configurable: true + /* ...other flags, probable "value:..." */ + }; + } + +}); + +alert( Object.keys(user) ); // a, b, c +``` + +Let's note once again: we only need to intercept `[[GetOwnProperty]]` if the property is absent in the object. + +## Protected properties with "deleteProperty" and other traps + +There's a widespread convention that properties and methods prefixed by an underscore `_` are internal. They shouldn't be accessed from outside the object. + +Technically that's possible though: + +```js run +let user = { + name: "John", + _password: "secret" +}; + +alert(user._password); // secret +``` + +Let's use proxies to prevent any access to properties starting with `_`. + +We'll need the traps: +- `get` to throw an error when reading such property, +- `set` to throw an error when writing, +- `deleteProperty` to throw an error when deleting, +- `ownKeys` to exclude properties starting with `_` from `for..in` and methods like `Object.keys`. + +Here's the code: + +```js run +let user = { + name: "John", + _password: "***" +}; + +user = new Proxy(user, { +*!* + get(target, prop) { +*/!* + if (prop.startsWith('_')) { + throw new Error("Access denied"); + } + let value = target[prop]; + return (typeof value === 'function') ? value.bind(target) : value; // (*) + }, +*!* + set(target, prop, val) { // to intercept property writing +*/!* + if (prop.startsWith('_')) { + throw new Error("Access denied"); + } else { + target[prop] = val; + return true; + } + }, +*!* + deleteProperty(target, prop) { // to intercept property deletion +*/!* + if (prop.startsWith('_')) { + throw new Error("Access denied"); + } else { + delete target[prop]; + return true; + } + }, +*!* + ownKeys(target) { // to intercept property list +*/!* + return Object.keys(target).filter(key => !key.startsWith('_')); + } +}); + +// "get" doesn't allow to read _password +try { + alert(user._password); // Error: Access denied +} catch(e) { alert(e.message); } + +// "set" doesn't allow to write _password +try { + user._password = "test"; // Error: Access denied +} catch(e) { alert(e.message); } + +// "deleteProperty" doesn't allow to delete _password +try { + delete user._password; // Error: Access denied +} catch(e) { alert(e.message); } + +// "ownKeys" filters out _password +for(let key in user) alert(key); // name +``` + +Please note the important detail in the `get` trap, in the line `(*)`: + +```js +get(target, prop) { + // ... + let value = target[prop]; +*!* + return (typeof value === 'function') ? value.bind(target) : value; // (*) +*/!* +} +``` + +Why do we need a function to call `value.bind(target)`? + +The reason is that object methods, such as `user.checkPassword()`, must be able to access `_password`: + +```js +user = { + // ... + checkPassword(value) { + // object method must be able to read _password + return value === this._password; + } +} +``` + + +A call to `user.checkPassword()` gets proxied `user` as `this` (the object before dot becomes `this`), so when it tries to access `this._password`, the `get` trap activates (it triggers on any property read) and throws an error. + +So we bind the context of object methods to the original object, `target`, in the line `(*)`. Then their future calls will use `target` as `this`, without any traps. + +That solution usually works, but isn't ideal, as a method may pass the unproxied object somewhere else, and then we'll get messed up: where's the original object, and where's the proxied one? + +Besides, an object may be proxied multiple times (multiple proxies may add different "tweaks" to the object), and if we pass an unwrapped object to a method, there may be unexpected consequences. + +So, such a proxy shouldn't be used everywhere. + +```smart header="Private properties of a class" +Modern JavaScript engines natively support private properties in classes, prefixed with `#`. They are described in the article <info:private-protected-properties-methods>. No proxies required. + +Such properties have their own issues though. In particular, they are not inherited. +``` + +## "In range" with "has" trap + +Let's see more examples. + +We have a range object: + +```js +let range = { + start: 1, + end: 10 +}; +``` + +We'd like to use the `in` operator to check that a number is in `range`. + +The `has` trap intercepts `in` calls. + +`has(target, property)` + +- `target` -- is the target object, passed as the first argument to `new Proxy`, +- `property` -- property name + +Here's the demo: + +```js run +let range = { + start: 1, + end: 10 +}; + +range = new Proxy(range, { +*!* + has(target, prop) { +*/!* + return prop >= target.start && prop <= target.end; + } +}); + +*!* +alert(5 in range); // true +alert(50 in range); // false +*/!* +``` + +Nice syntactic sugar, isn't it? And very simple to implement. + +## Wrapping functions: "apply" [#proxy-apply] + +We can wrap a proxy around a function as well. + +The `apply(target, thisArg, args)` trap handles calling a proxy as function: + +- `target` is the target object (function is an object in JavaScript), +- `thisArg` is the value of `this`. +- `args` is a list of arguments. + +For example, let's recall `delay(f, ms)` decorator, that we did in the article <info:call-apply-decorators>. + +In that article we did it without proxies. A call to `delay(f, ms)` returned a function that forwards all calls to `f` after `ms` milliseconds. + +Here's the previous, function-based implementation: + +```js run +function delay(f, ms) { + // return a wrapper that passes the call to f after the timeout + return function() { // (*) + setTimeout(() => f.apply(this, arguments), ms); + }; +} + +function sayHi(user) { + alert(`Hello, ${user}!`); +} + +// after this wrapping, calls to sayHi will be delayed for 3 seconds +sayHi = delay(sayHi, 3000); + +sayHi("John"); // Hello, John! (after 3 seconds) +``` + +As we've seen already, that mostly works. The wrapper function `(*)` performs the call after the timeout. + +But a wrapper function does not forward property read/write operations or anything else. After the wrapping, the access is lost to properties of the original functions, such as `name`, `length` and others: + +```js run +function delay(f, ms) { + return function() { + setTimeout(() => f.apply(this, arguments), ms); + }; +} + +function sayHi(user) { + alert(`Hello, ${user}!`); +} + +*!* +alert(sayHi.length); // 1 (function length is the arguments count in its declaration) +*/!* + +sayHi = delay(sayHi, 3000); + +*!* +alert(sayHi.length); // 0 (in the wrapper declaration, there are zero arguments) +*/!* +``` + +`Proxy` is much more powerful, as it forwards everything to the target object. + +Let's use `Proxy` instead of a wrapping function: + +```js run +function delay(f, ms) { + return new Proxy(f, { + apply(target, thisArg, args) { + setTimeout(() => target.apply(thisArg, args), ms); + } + }); +} + +function sayHi(user) { + alert(`Hello, ${user}!`); +} + +sayHi = delay(sayHi, 3000); + +*!* +alert(sayHi.length); // 1 (*) proxy forwards "get length" operation to the target +*/!* + +sayHi("John"); // Hello, John! (after 3 seconds) +``` + +The result is the same, but now not only calls, but all operations on the proxy are forwarded to the original function. So `sayHi.length` is returned correctly after the wrapping in the line `(*)`. + +We've got a "richer" wrapper. + +Other traps exist: the full list is in the beginning of this article. Their usage pattern is similar to the above. + +## Reflect + +`Reflect` is a built-in object that simplifies creation of `Proxy`. + +It was said previously that internal methods, such as `[[Get]]`, `[[Set]]` and others are specification-only, they can't be called directly. + +The `Reflect` object makes that somewhat possible. Its methods are minimal wrappers around the internal methods. + +Here are examples of operations and `Reflect` calls that do the same: + +| Operation | `Reflect` call | Internal method | +|-----------------|----------------|-------------| +| `obj[prop]` | `Reflect.get(obj, prop)` | `[[Get]]` | +| `obj[prop] = value` | `Reflect.set(obj, prop, value)` | `[[Set]]` | +| `delete obj[prop]` | `Reflect.deleteProperty(obj, prop)` | `[[Delete]]` | +| `new F(value)` | `Reflect.construct(F, value)` | `[[Construct]]` | +| ... | ... | ... | + +For example: + +```js run +let user = {}; + +Reflect.set(user, 'name', 'John'); + +alert(user.name); // John +``` + +In particular, `Reflect` allows us to call operators (`new`, `delete`...) as functions (`Reflect.construct`, `Reflect.deleteProperty`, ...). That's an interesting capability, but here another thing is important. + +**For every internal method, trappable by `Proxy`, there's a corresponding method in `Reflect`, with the same name and arguments as the `Proxy` trap.** + +So we can use `Reflect` to forward an operation to the original object. + +In this example, both traps `get` and `set` transparently (as if they didn't exist) forward reading/writing operations to the object, showing a message: + +```js run +let user = { + name: "John", +}; + +user = new Proxy(user, { + get(target, prop, receiver) { + alert(`GET ${prop}`); +*!* + return Reflect.get(target, prop, receiver); // (1) +*/!* + }, + set(target, prop, val, receiver) { + alert(`SET ${prop}=${val}`); +*!* + return Reflect.set(target, prop, val, receiver); // (2) +*/!* + } +}); + +let name = user.name; // shows "GET name" +user.name = "Pete"; // shows "SET name=Pete" +``` + +Here: + +- `Reflect.get` reads an object property. +- `Reflect.set` writes an object property and returns `true` if successful, `false` otherwise. + +That is, everything's simple: if a trap wants to forward the call to the object, it's enough to call `Reflect.<method>` with the same arguments. + +In most cases we can do the same without `Reflect`, for instance, reading a property `Reflect.get(target, prop, receiver)` can be replaced by `target[prop]`. There are important nuances though. + +### Proxying a getter + +Let's see an example that demonstrates why `Reflect.get` is better. And we'll also see why `get/set` have the third argument `receiver`, that we didn't use before. + +We have an object `user` with `_name` property and a getter for it. + +Here's a proxy around it: + +```js run +let user = { + _name: "Guest", + get name() { + return this._name; + } +}; + +*!* +let userProxy = new Proxy(user, { + get(target, prop, receiver) { + return target[prop]; + } +}); +*/!* + +alert(userProxy.name); // Guest +``` + +The `get` trap is "transparent" here, it returns the original property, and doesn't do anything else. That's enough for our example. + +Everything seems to be all right. But let's make the example a little bit more complex. + +After inheriting another object `admin` from `user`, we can observe the incorrect behavior: + +```js run +let user = { + _name: "Guest", + get name() { + return this._name; + } +}; + +let userProxy = new Proxy(user, { + get(target, prop, receiver) { + return target[prop]; // (*) target = user + } +}); + +*!* +let admin = { + __proto__: userProxy, + _name: "Admin" +}; + +// Expected: Admin +alert(admin.name); // outputs: Guest (?!?) +*/!* +``` + +Reading `admin.name` should return `"Admin"`, not `"Guest"`! + +What's the matter? Maybe we did something wrong with the inheritance? + +But if we remove the proxy, then everything will work as expected. + +The problem is actually in the proxy, in the line `(*)`. + +1. When we read `admin.name`, as `admin` object doesn't have such own property, the search goes to its prototype. +2. The prototype is `userProxy`. +3. When reading `name` property from the proxy, its `get` trap triggers and returns it from the original object as `target[prop]` in the line `(*)`. + + A call to `target[prop]`, when `prop` is a getter, runs its code in the context `this=target`. So the result is `this._name` from the original object `target`, that is: from `user`. + +To fix such situations, we need `receiver`, the third argument of `get` trap. It keeps the correct `this` to be passed to a getter. In our case that's `admin`. + +How to pass the context for a getter? For a regular function we could use `call/apply`, but that's a getter, it's not "called", just accessed. + +`Reflect.get` can do that. Everything will work right if we use it. + +Here's the corrected variant: + +```js run +let user = { + _name: "Guest", + get name() { + return this._name; + } +}; + +let userProxy = new Proxy(user, { + get(target, prop, receiver) { // receiver = admin +*!* + return Reflect.get(target, prop, receiver); // (*) +*/!* + } +}); + + +let admin = { + __proto__: userProxy, + _name: "Admin" +}; + +*!* +alert(admin.name); // Admin +*/!* +``` + +Now `receiver` that keeps a reference to the correct `this` (that is `admin`), is passed to the getter using `Reflect.get` in the line `(*)`. + +We can rewrite the trap even shorter: + +```js +get(target, prop, receiver) { + return Reflect.get(*!*...arguments*/!*); +} +``` + + +`Reflect` calls are named exactly the same way as traps and accept the same arguments. They were specifically designed this way. + +So, `return Reflect...` provides a safe no-brainer to forward the operation and make sure we don't forget anything related to that. + +## Proxy limitations + +Proxies provide a unique way to alter or tweak the behavior of the existing objects at the lowest level. Still, it's not perfect. There are limitations. + +### Built-in objects: Internal slots + +Many built-in objects, for example `Map`, `Set`, `Date`, `Promise` and others make use of so-called "internal slots". + +These are like properties, but reserved for internal, specification-only purposes. For instance, `Map` stores items in the internal slot `[[MapData]]`. Built-in methods access them directly, not via `[[Get]]/[[Set]]` internal methods. So `Proxy` can't intercept that. + +Why care? They're internal anyway! + +Well, here's the issue. After a built-in object like that gets proxied, the proxy doesn't have these internal slots, so built-in methods will fail. + +For example: + +```js run +let map = new Map(); + +let proxy = new Proxy(map, {}); + +*!* +proxy.set('test', 1); // Error +*/!* +``` + +Internally, a `Map` stores all data in its `[[MapData]]` internal slot. The proxy doesn't have such a slot. The [built-in method `Map.prototype.set`](https://tc39.es/ecma262/#sec-map.prototype.set) method tries to access the internal property `this.[[MapData]]`, but because `this=proxy`, can't find it in `proxy` and just fails. + +Fortunately, there's a way to fix it: + +```js run +let map = new Map(); + +let proxy = new Proxy(map, { + get(target, prop, receiver) { + let value = Reflect.get(...arguments); +*!* + return typeof value == 'function' ? value.bind(target) : value; +*/!* + } +}); + +proxy.set('test', 1); +alert(proxy.get('test')); // 1 (works!) +``` + +Now it works fine, because `get` trap binds function properties, such as `map.set`, to the target object (`map`) itself. + +Unlike the previous example, the value of `this` inside `proxy.set(...)` will be not `proxy`, but the original `map`. So when the internal implementation of `set` tries to access `this.[[MapData]]` internal slot, it succeeds. + +```smart header="`Array` has no internal slots" +A notable exception: built-in `Array` doesn't use internal slots. That's for historical reasons, as it appeared so long ago. + +So there's no such problem when proxying an array. +``` + +### Private fields + +A similar thing happens with private class fields. + +For example, `getName()` method accesses the private `#name` property and breaks after proxying: + +```js run +class User { + #name = "Guest"; + + getName() { + return this.#name; + } +} + +let user = new User(); + +user = new Proxy(user, {}); + +*!* +alert(user.getName()); // Error +*/!* +``` + +The reason is that private fields are implemented using internal slots. JavaScript does not use `[[Get]]/[[Set]]` when accessing them. + +In the call `getName()` the value of `this` is the proxied `user`, and it doesn't have the slot with private fields. + +Once again, the solution with binding the method makes it work: + +```js run +class User { + #name = "Guest"; + + getName() { + return this.#name; + } +} + +let user = new User(); + +user = new Proxy(user, { + get(target, prop, receiver) { + let value = Reflect.get(...arguments); + return typeof value == 'function' ? value.bind(target) : value; + } +}); + +alert(user.getName()); // Guest +``` + +That said, the solution has drawbacks, as explained previously: it exposes the original object to the method, potentially allowing it to be passed further and breaking other proxied functionality. + +### Proxy != target + +The proxy and the original object are different objects. That's natural, right? + +So if we use the original object as a key, and then proxy it, then the proxy can't be found: + +```js run +let allUsers = new Set(); + +class User { + constructor(name) { + this.name = name; + allUsers.add(this); + } +} + +let user = new User("John"); + +alert(allUsers.has(user)); // true + +user = new Proxy(user, {}); + +*!* +alert(allUsers.has(user)); // false +*/!* +``` + +As we can see, after proxying we can't find `user` in the set `allUsers`, because the proxy is a different object. + +```warn header="Proxies can't intercept a strict equality test `===`" +Proxies can intercept many operators, such as `new` (with `construct`), `in` (with `has`), `delete` (with `deleteProperty`) and so on. + +But there's no way to intercept a strict equality test for objects. An object is strictly equal to itself only, and no other value. + +So all operations and built-in classes that compare objects for equality will differentiate between the object and the proxy. No transparent replacement here. +``` + +## Revocable proxies + +A *revocable* proxy is a proxy that can be disabled. + +Let's say we have a resource, and would like to close access to it any moment. + +What we can do is to wrap it into a revocable proxy, without any traps. Such a proxy will forward operations to object, and we can disable it at any moment. + +The syntax is: + +```js +let {proxy, revoke} = Proxy.revocable(target, handler) +``` + +The call returns an object with the `proxy` and `revoke` function to disable it. + +Here's an example: + +```js run +let object = { + data: "Valuable data" +}; + +let {proxy, revoke} = Proxy.revocable(object, {}); + +// pass the proxy somewhere instead of object... +alert(proxy.data); // Valuable data + +// later in our code +revoke(); + +// the proxy isn't working any more (revoked) +alert(proxy.data); // Error +``` + +A call to `revoke()` removes all internal references to the target object from the proxy, so they are no longer connected. + +Initially, `revoke` is separate from `proxy`, so that we can pass `proxy` around while leaving `revoke` in the current scope. + +We can also bind `revoke` method to proxy by setting `proxy.revoke = revoke`. + +Another option is to create a `WeakMap` that has `proxy` as the key and the corresponding `revoke` as the value, that allows to easily find `revoke` for a proxy: + +```js run +*!* +let revokes = new WeakMap(); +*/!* + +let object = { + data: "Valuable data" +}; + +let {proxy, revoke} = Proxy.revocable(object, {}); + +revokes.set(proxy, revoke); + +// ..somewhere else in our code.. +revoke = revokes.get(proxy); +revoke(); + +alert(proxy.data); // Error (revoked) +``` + +We use `WeakMap` instead of `Map` here because it won't block garbage collection. If a proxy object becomes "unreachable" (e.g. no variable references it any more), `WeakMap` allows it to be wiped from memory together with its `revoke` that we won't need any more. + +## References + +- Specification: [Proxy](https://tc39.es/ecma262/#sec-proxy-object-internal-methods-and-internal-slots). +- MDN: [Proxy](mdn:/JavaScript/Reference/Global_Objects/Proxy). + +## Summary + +`Proxy` is a wrapper around an object, that forwards operations on it to the object, optionally trapping some of them. + +It can wrap any kind of object, including classes and functions. + +The syntax is: + +```js +let proxy = new Proxy(target, { + /* traps */ +}); +``` + +...Then we should use `proxy` everywhere instead of `target`. A proxy doesn't have its own properties or methods. It traps an operation if the trap is provided, otherwise forwards it to `target` object. + +We can trap: +- Reading (`get`), writing (`set`), deleting (`deleteProperty`) a property (even a non-existing one). +- Calling a function (`apply` trap). +- The `new` operator (`construct` trap). +- Many other operations (the full list is at the beginning of the article and in the [docs](mdn:/JavaScript/Reference/Global_Objects/Proxy)). + +That allows us to create "virtual" properties and methods, implement default values, observable objects, function decorators and so much more. + +We can also wrap an object multiple times in different proxies, decorating it with various aspects of functionality. + +The [Reflect](mdn:/JavaScript/Reference/Global_Objects/Reflect) API is designed to complement [Proxy](mdn:/JavaScript/Reference/Global_Objects/Proxy). For any `Proxy` trap, there's a `Reflect` call with same arguments. We should use those to forward calls to target objects. + +Proxies have some limitations: + +- Built-in objects have "internal slots", access to those can't be proxied. See the workaround above. +- The same holds true for private class fields, as they are internally implemented using slots. So proxied method calls must have the target object as `this` to access them. +- Object equality tests `===` can't be intercepted. +- Performance: benchmarks depend on an engine, but generally accessing a property using a simplest proxy takes a few times longer. In practice that only matters for some "bottleneck" objects though. diff --git a/1-js/99-js-misc/01-proxy/proxy-inherit-admin.svg b/1-js/99-js-misc/01-proxy/proxy-inherit-admin.svg new file mode 100644 index 000000000..3fba64606 --- /dev/null +++ b/1-js/99-js-misc/01-proxy/proxy-inherit-admin.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="241" height="285" viewBox="0 0 241 285"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="misc" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="proxy-inherit-admin.svg"><path id="Oval" stroke="#DBAF88" stroke-dasharray="5" stroke-width="2" d="M116 181c43.63 0 79-30.668 79-68.5S159.63 44 116 44s-79 30.668-79 68.5S72.37 181 116 181z"/><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M52 96h130v48H52z"/><text id="_name:-"Guest"-name:" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="59" y="116">_name: "Guest"</tspan> <tspan x="59" y="131">name: getter</tspan></text><path id="Rectangle-2-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M52 242h130v28H52z"/><text id="_name:-"Admin"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="59" y="260">_name: "Admin"</tspan></text><text id="user-(proxied)" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="57" y="29">user (proxied)</tspan></text><text id="original-user" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="61" y="87">original user</tspan></text><path id="Line-Copy-3" fill="#C06334" fill-rule="nonzero" d="M115.963 180.5l9.5 19h-8V241h-3v-41.5h-8l9.5-19z" transform="matrix(-1 0 0 1 232 0)"/><text id="admin" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="51" y="233">admin</tspan></text><text id="[[Prototype]]" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="122" y="217">[[Prototype]]</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/99-js-misc/01-proxy/proxy-inherit.svg b/1-js/99-js-misc/01-proxy/proxy-inherit.svg new file mode 100644 index 000000000..6c34c0f4e --- /dev/null +++ b/1-js/99-js-misc/01-proxy/proxy-inherit.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="230" height="206" viewBox="0 0 230 206"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="misc" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="proxy-inherit.svg"><path id="Oval" stroke="#DBAF88" stroke-dasharray="5" stroke-width="2" d="M116 181c43.63 0 79-30.668 79-68.5S159.63 44 116 44s-79 30.668-79 68.5S72.37 181 116 181z"/><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M52 96h130v48H52z"/><text id="_name:-"Guest"-name:" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="59" y="116">_name: "Guest"</tspan> <tspan x="59" y="131">name: getter</tspan></text><text id="user-(proxied)" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="57" y="29">user (proxied)</tspan></text><text id="original-user" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="61" y="87">original user</tspan></text></g></g></svg> \ No newline at end of file diff --git a/1-js/99-js-misc/01-proxy/proxy.svg b/1-js/99-js-misc/01-proxy/proxy.svg new file mode 100644 index 000000000..6b2224cfd --- /dev/null +++ b/1-js/99-js-misc/01-proxy/proxy.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="292" height="180" viewBox="0 0 292 180"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="misc" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="proxy.svg"><path id="Rectangle-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M191 81h73v26h-73z"/><text id="test:-5" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="198" y="98">test: 5</tspan></text><text id="proxy" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="186" y="17">proxy</tspan></text><text id="target" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="198" y="74">target</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M116.537 72.5l19 9.5-19 9.5-.001-8H4.463v-3h112.073v-8z"/><path id="Line-Copy-4" fill="#C06334" fill-rule="nonzero" d="M22 98.5v8h114.574v3H22v8L3 108l19-9.5z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M190.478 89l-14-7v14l14-7zM146 88v2h-4v-2h4zm7 0v2h-4v-2h4zm7 0v2h-4v-2h4zm7 0v2h-4v-2h4zm7 0v2h-4v-2h4z"/><path id="Line-Copy-2" fill="#C06334" fill-rule="nonzero" d="M157 93v6h2v2h-2v6l-14-7 14-7zm9 6v2h-4v-2h4zm7 0v2h-4v-2h4zm7 0v2h-4v-2h4zm7 0v2h-4v-2h4z"/><text id="get-proxy.test" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="7" y="70">get proxy.test</tspan></text><text id="5" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="71" y="128">5</tspan></text><path id="Oval" stroke="#DBAF88" stroke-dasharray="5" stroke-width="2" d="M207 160c36.45 0 66-29.55 66-66s-29.55-66-66-66-66 29.55-66 66 29.55 66 66 66z"/></g></g></svg> \ No newline at end of file diff --git a/1-js/99-js-misc/02-eval/1-eval-calculator/solution.md b/1-js/99-js-misc/02-eval/1-eval-calculator/solution.md new file mode 100644 index 000000000..24d40c9b8 --- /dev/null +++ b/1-js/99-js-misc/02-eval/1-eval-calculator/solution.md @@ -0,0 +1,11 @@ +Let's use `eval` to calculate the maths expression: + +```js demo run +let expr = prompt("Type an arithmetic expression?", '2*3+2'); + +alert( eval(expr) ); +``` + +The user can input any text or code though. + +To make things safe, and limit it to arithmetics only, we can check the `expr` using a [regular expression](info:regular-expressions), so that it only may contain digits and operators. diff --git a/1-js/99-js-misc/02-eval/1-eval-calculator/task.md b/1-js/99-js-misc/02-eval/1-eval-calculator/task.md new file mode 100644 index 000000000..ece43ec9e --- /dev/null +++ b/1-js/99-js-misc/02-eval/1-eval-calculator/task.md @@ -0,0 +1,11 @@ +importance: 4 + +--- + +# Eval-calculator + +Create a calculator that prompts for an arithmetic expression and returns its result. + +There's no need to check the expression for correctness in this task. Just evaluate and return the result. + +[demo] diff --git a/1-js/99-js-misc/02-eval/article.md b/1-js/99-js-misc/02-eval/article.md new file mode 100644 index 000000000..1fdafeeec --- /dev/null +++ b/1-js/99-js-misc/02-eval/article.md @@ -0,0 +1,114 @@ +# Eval: run a code string + +The built-in `eval` function allows to execute a string of code. + +The syntax is: + +```js +let result = eval(code); +``` + +For example: + +```js run +let code = 'alert("Hello")'; +eval(code); // Hello +``` + +A string of code may be long, contain line breaks, function declarations, variables and so on. + +The result of `eval` is the result of the last statement. + +For example: +```js run +let value = eval('1+1'); +alert(value); // 2 +``` + +```js run +let value = eval('let i = 0; ++i'); +alert(value); // 1 +``` + +The eval'ed code is executed in the current lexical environment, so it can see outer variables: + +```js run no-beautify +let a = 1; + +function f() { + let a = 2; + +*!* + eval('alert(a)'); // 2 +*/!* +} + +f(); +``` + +It can change outer variables as well: + +```js untrusted refresh run +let x = 5; +eval("x = 10"); +alert(x); // 10, value modified +``` + +In strict mode, `eval` has its own lexical environment. So functions and variables, declared inside eval, are not visible outside: + +```js untrusted refresh run +// reminder: 'use strict' is enabled in runnable examples by default + +eval("let x = 5; function f() {}"); + +alert(typeof x); // undefined (no such variable) +// function f is also not visible +``` + +Without `use strict`, `eval` doesn't have its own lexical environment, so we would see `x` and `f` outside. + +## Using "eval" + +In modern programming `eval` is used very sparingly. It's often said that "eval is evil". + +The reason is simple: long, long time ago JavaScript was a much weaker language, many things could only be done with `eval`. But that time passed a decade ago. + +Right now, there's almost no reason to use `eval`. If someone is using it, there's a good chance they can replace it with a modern language construct or a [JavaScript Module](info:modules). + +Please note that its ability to access outer variables has side-effects. + +Code minifiers (tools used before JS gets to production, to compress it) rename local variables into shorter ones (like `a`, `b` etc) to make the code smaller. That's usually safe, but not if `eval` is used, as local variables may be accessed from eval'ed code string. So minifiers don't do that renaming for all variables potentially visible from `eval`. That negatively affects code compression ratio. + +Using outer local variables inside `eval` is also considered a bad programming practice, as it makes maintaining the code more difficult. + +There are two ways how to be totally safe from such problems. + +**If eval'ed code doesn't use outer variables, please call `eval` as `window.eval(...)`:** + +This way the code is executed in the global scope: + +```js untrusted refresh run +let x = 1; +{ + let x = 5; + window.eval('alert(x)'); // 1 (global variable) +} +``` + +**If eval'ed code needs local variables, change `eval` to `new Function` and pass them as arguments:** + +```js run +let f = new Function('a', 'alert(a)'); + +f(5); // 5 +``` + +The `new Function` construct is explained in the chapter <info:new-function>. It creates a function from a string, also in the global scope. So it can't see local variables. But it's so much clearer to pass them explicitly as arguments, like in the example above. + +## Summary + +A call to `eval(code)` runs the string of code and returns the result of the last statement. +- Rarely used in modern JavaScript, as there's usually no need. +- Can access outer local variables. That's considered bad practice. +- Instead, to `eval` the code in the global scope, use `window.eval(code)`. +- Or, if your code needs some data from the outer scope, use `new Function` and pass it as arguments. diff --git a/1-js/99-js-misc/03-currying-partials/article.md b/1-js/99-js-misc/03-currying-partials/article.md new file mode 100644 index 000000000..d71ac23f8 --- /dev/null +++ b/1-js/99-js-misc/03-currying-partials/article.md @@ -0,0 +1,188 @@ +libs: + - lodash + +--- + +# Currying + +[Currying](https://en.wikipedia.org/wiki/Currying) is an advanced technique of working with functions. It's used not only in JavaScript, but in other languages as well. + +Currying is a transformation of functions that translates a function from callable as `f(a, b, c)` into callable as `f(a)(b)(c)`. + +Currying doesn't call a function. It just transforms it. + +Let's see an example first, to better understand what we're talking about, and then practical applications. + +We'll create a helper function `curry(f)` that performs currying for a two-argument `f`. In other words, `curry(f)` for two-argument `f(a, b)` translates it into a function that runs as `f(a)(b)`: + +```js run +*!* +function curry(f) { // curry(f) does the currying transform + return function(a) { + return function(b) { + return f(a, b); + }; + }; +} +*/!* + +// usage +function sum(a, b) { + return a + b; +} + +let curriedSum = curry(sum); + +alert( curriedSum(1)(2) ); // 3 +``` + +As you can see, the implementation is straightforward: it's just two wrappers. + +- The result of `curry(func)` is a wrapper `function(a)`. +- When it is called like `curriedSum(1)`, the argument is saved in the Lexical Environment, and a new wrapper is returned `function(b)`. +- Then this wrapper is called with `2` as an argument, and it passes the call to the original `sum`. + +More advanced implementations of currying, such as [_.curry](https://lodash.com/docs#curry) from lodash library, return a wrapper that allows a function to be called both normally and partially: + +```js run +function sum(a, b) { + return a + b; +} + +let curriedSum = _.curry(sum); // using _.curry from lodash library + +alert( curriedSum(1, 2) ); // 3, still callable normally +alert( curriedSum(1)(2) ); // 3, called partially +``` + +## Currying? What for? + +To understand the benefits we need a worthy real-life example. + +For instance, we have the logging function `log(date, importance, message)` that formats and outputs the information. In real projects such functions have many useful features like sending logs over the network, here we'll just use `alert`: + +```js +function log(date, importance, message) { + alert(`[${date.getHours()}:${date.getMinutes()}] [${importance}] ${message}`); +} +``` + +Let's curry it! + +```js +log = _.curry(log); +``` + +After that `log` works normally: + +```js +log(new Date(), "DEBUG", "some debug"); // log(a, b, c) +``` + +...But also works in the curried form: + +```js +log(new Date())("DEBUG")("some debug"); // log(a)(b)(c) +``` + +Now we can easily make a convenience function for current logs: + +```js +// logNow will be the partial of log with fixed first argument +let logNow = log(new Date()); + +// use it +logNow("INFO", "message"); // [HH:mm] INFO message +``` + +Now `logNow` is `log` with fixed first argument, in other words "partially applied function" or "partial" for short. + +We can go further and make a convenience function for current debug logs: + +```js +let debugNow = logNow("DEBUG"); + +debugNow("message"); // [HH:mm] DEBUG message +``` + +So: +1. We didn't lose anything after currying: `log` is still callable normally. +2. We can easily generate partial functions such as for today's logs. + +## Advanced curry implementation + +In case you'd like to get in to the details, here's the "advanced" curry implementation for multi-argument functions that we could use above. + +It's pretty short: + +```js +function curry(func) { + + return function curried(...args) { + if (args.length >= func.length) { + return func.apply(this, args); + } else { + return function(...args2) { + return curried.apply(this, args.concat(args2)); + } + } + }; + +} +``` + +Usage examples: + +```js +function sum(a, b, c) { + return a + b + c; +} + +let curriedSum = curry(sum); + +alert( curriedSum(1, 2, 3) ); // 6, still callable normally +alert( curriedSum(1)(2,3) ); // 6, currying of 1st arg +alert( curriedSum(1)(2)(3) ); // 6, full currying +``` + +The new `curry` may look complicated, but it's actually easy to understand. + +The result of `curry(func)` call is the wrapper `curried` that looks like this: + +```js +// func is the function to transform +function curried(...args) { + if (args.length >= func.length) { // (1) + return func.apply(this, args); + } else { + return function(...args2) { // (2) + return curried.apply(this, args.concat(args2)); + } + } +}; +``` + +When we run it, there are two `if` execution branches: + +1. If passed `args` count is the same or more than the original function has in its definition (`func.length`) , then just pass the call to it using `func.apply`. +2. Otherwise, get a partial: we don't call `func` just yet. Instead, another wrapper is returned, that will re-apply `curried` providing previous arguments together with the new ones. + +Then, if we call it, again, we'll get either a new partial (if not enough arguments) or, finally, the result. + +```smart header="Fixed-length functions only" +The currying requires the function to have a fixed number of arguments. + +A function that uses rest parameters, such as `f(...args)`, can't be curried this way. +``` + +```smart header="A little more than currying" +By definition, currying should convert `sum(a, b, c)` into `sum(a)(b)(c)`. + +But most implementations of currying in JavaScript are advanced, as described: they also keep the function callable in the multi-argument variant. +``` + +## Summary + +*Currying* is a transform that makes `f(a,b,c)` callable as `f(a)(b)(c)`. JavaScript implementations usually both keep the function callable normally and return the partial if the arguments count is not enough. + +Currying allows us to easily get partials. As we've seen in the logging example, after currying the three argument universal function `log(date, importance, message)` gives us partials when called with one argument (like `log(date)`) or two arguments (like `log(date, importance)`). diff --git a/1-js/04-object-basics/04-object-methods/2-check-syntax/solution.md b/1-js/99-js-misc/04-reference-type/2-check-syntax/solution.md similarity index 61% rename from 1-js/04-object-basics/04-object-methods/2-check-syntax/solution.md rename to 1-js/99-js-misc/04-reference-type/2-check-syntax/solution.md index e2e87de7c..ba5d3bf04 100644 --- a/1-js/04-object-basics/04-object-methods/2-check-syntax/solution.md +++ b/1-js/99-js-misc/04-reference-type/2-check-syntax/solution.md @@ -11,17 +11,17 @@ let user = { (user.go)() // error! ``` -The error message in most browsers does not give understanding what went wrong. +The error message in most browsers does not give us much of a clue about what went wrong. **The error appears because a semicolon is missing after `user = {...}`.** -JavaScript does not assume a semicolon before a bracket `(user.go)()`, so it reads the code like: +JavaScript does not auto-insert a semicolon before a bracket `(user.go)()`, so it reads the code like: ```js no-beautify let user = { go:... }(user.go)() ``` -Then we can also see that such a joint expression is syntactically a call of the object `{ go: ... }` as a function with the argument `(user.go)`. And that also happens on the same line with `let user`, so the `user` object has not yet even been defined, hence the error. +Then we can also see that such a joint expression is syntactically a call of the object `{ go: ... }` as a function with the argument `(user.go)`. And that also happens on the same line with `let user`, so the `user` object has not yet even been defined, hence the error. If we insert the semicolon, all is fine: @@ -34,10 +34,4 @@ let user = { (user.go)() // John ``` -Please note that brackets around `(user.go)` do nothing here. Usually they setup the order of operations, but here the dot `.` works first anyway, so there's no effect. Only the semicolon thing matters. - - - - - - +Please note that parentheses around `(user.go)` do nothing here. Usually they setup the order of operations, but here the dot `.` works first anyway, so there's no effect. Only the semicolon thing matters. diff --git a/1-js/04-object-basics/04-object-methods/2-check-syntax/task.md b/1-js/99-js-misc/04-reference-type/2-check-syntax/task.md similarity index 100% rename from 1-js/04-object-basics/04-object-methods/2-check-syntax/task.md rename to 1-js/99-js-misc/04-reference-type/2-check-syntax/task.md diff --git a/1-js/04-object-basics/04-object-methods/3-why-this/solution.md b/1-js/99-js-misc/04-reference-type/3-why-this/solution.md similarity index 65% rename from 1-js/04-object-basics/04-object-methods/3-why-this/solution.md rename to 1-js/99-js-misc/04-reference-type/3-why-this/solution.md index 89bc0d722..e4ee78748 100644 --- a/1-js/04-object-basics/04-object-methods/3-why-this/solution.md +++ b/1-js/99-js-misc/04-reference-type/3-why-this/solution.md @@ -3,9 +3,9 @@ Here's the explanations. 1. That's a regular object method call. -2. The same, brackets do not change the order of operations here, the dot is first anyway. +2. The same, parentheses do not change the order of operations here, the dot is first anyway. -3. Here we have a more complex call `(expression).method()`. The call works as if it were split into two lines: +3. Here we have a more complex call `(expression)()`. The call works as if it were split into two lines: ```js no-beautify f = obj.go; // calculate the expression @@ -14,7 +14,7 @@ Here's the explanations. Here `f()` is executed as a function, without `this`. -4. The similar thing as `(3)`, to the left of the dot `.` we have an expression. +4. The similar thing as `(3)`, to the left of the parentheses `()` we have an expression. To explain the behavior of `(3)` and `(4)` we need to recall that property accessors (dot or square brackets) return a value of the Reference Type. diff --git a/1-js/04-object-basics/04-object-methods/3-why-this/task.md b/1-js/99-js-misc/04-reference-type/3-why-this/task.md similarity index 84% rename from 1-js/04-object-basics/04-object-methods/3-why-this/task.md rename to 1-js/99-js-misc/04-reference-type/3-why-this/task.md index f22de29cc..e2c073f62 100644 --- a/1-js/04-object-basics/04-object-methods/3-why-this/task.md +++ b/1-js/99-js-misc/04-reference-type/3-why-this/task.md @@ -4,7 +4,7 @@ importance: 3 # Explain the value of "this" -In the code below we intend to call `user.go()` method 4 times in a row. +In the code below we intend to call `obj.go()` method 4 times in a row. But calls `(1)` and `(2)` works differently from `(3)` and `(4)`. Why? diff --git a/1-js/99-js-misc/04-reference-type/article.md b/1-js/99-js-misc/04-reference-type/article.md new file mode 100644 index 000000000..894db8fc6 --- /dev/null +++ b/1-js/99-js-misc/04-reference-type/article.md @@ -0,0 +1,108 @@ + +# Reference Type + +```warn header="In-depth language feature" +This article covers an advanced topic, to understand certain edge-cases better. + +It's not important. Many experienced developers live fine without knowing it. Read on if you want to know how things work under the hood. +``` + +A dynamically evaluated method call can lose `this`. + +For instance: + +```js run +let user = { + name: "John", + hi() { alert(this.name); }, + bye() { alert("Bye"); } +}; + +user.hi(); // works + +// now let's call user.hi or user.bye depending on the name +*!* +(user.name == "John" ? user.hi : user.bye)(); // Error! +*/!* +``` + +On the last line there is a conditional operator that chooses either `user.hi` or `user.bye`. In this case the result is `user.hi`. + +Then the method is immediately called with parentheses `()`. But it doesn't work correctly! + +As you can see, the call results in an error, because the value of `"this"` inside the call becomes `undefined`. + +This works (object dot method): +```js +user.hi(); +``` + +This doesn't (evaluated method): +```js +(user.name == "John" ? user.hi : user.bye)(); // Error! +``` + +Why? If we want to understand why it happens, let's get under the hood of how `obj.method()` call works. + +## Reference type explained + +Looking closely, we may notice two operations in `obj.method()` statement: + +1. First, the dot `'.'` retrieves the property `obj.method`. +2. Then parentheses `()` execute it. + +So, how does the information about `this` get passed from the first part to the second one? + +If we put these operations on separate lines, then `this` will be lost for sure: + +```js run +let user = { + name: "John", + hi() { alert(this.name); } +}; + +*!* +// split getting and calling the method in two lines +let hi = user.hi; +hi(); // Error, because this is undefined +*/!* +``` + +Here `hi = user.hi` puts the function into the variable, and then on the last line it is completely standalone, and so there's no `this`. + +**To make `user.hi()` calls work, JavaScript uses a trick -- the dot `'.'` returns not a function, but a value of the special [Reference Type](https://tc39.github.io/ecma262/#sec-reference-specification-type).** + +The Reference Type is a "specification type". We can't explicitly use it, but it is used internally by the language. + +The value of Reference Type is a three-value combination `(base, name, strict)`, where: + +- `base` is the object. +- `name` is the property name. +- `strict` is true if `use strict` is in effect. + +The result of a property access `user.hi` is not a function, but a value of Reference Type. For `user.hi` in strict mode it is: + +```js +// Reference Type value +(user, "hi", true) +``` + +When parentheses `()` are called on the Reference Type, they receive the full information about the object and its method, and can set the right `this` (`user` in this case). + +Reference type is a special "intermediary" internal type, with the purpose to pass information from dot `.` to calling parentheses `()`. + +Any other operation like assignment `hi = user.hi` discards the reference type as a whole, takes the value of `user.hi` (a function) and passes it on. So any further operation "loses" `this`. + +So, as the result, the value of `this` is only passed the right way if the function is called directly using a dot `obj.method()` or square brackets `obj['method']()` syntax (they do the same here). There are various ways to solve this problem such as [func.bind()](/bind#solution-2-bind). + +## Summary + +Reference Type is an internal type of the language. + +Reading a property, such as with dot `.` in `obj.method()` returns not exactly the property value, but a special "reference type" value that stores both the property value and the object it was taken from. + +That's for the subsequent method call `()` to get the object and set `this` to it. + +For all other operations, the reference type automatically becomes the property value (a function in our case). + +The whole mechanics is hidden from our eyes. It only matters in subtle cases, such as when a method is obtained dynamically from the object, using an expression. diff --git a/1-js/99-js-misc/05-bigint/article.md b/1-js/99-js-misc/05-bigint/article.md new file mode 100644 index 000000000..2a1cfc843 --- /dev/null +++ b/1-js/99-js-misc/05-bigint/article.md @@ -0,0 +1,130 @@ +# BigInt + +[recent caniuse="bigint"] + +`BigInt` is a special numeric type that provides support for integers of arbitrary length. + +A bigint is created by appending `n` to the end of an integer literal or by calling the function `BigInt` that creates bigints from strings, numbers etc. + +```js +const bigint = 1234567890123456789012345678901234567890n; + +const sameBigint = BigInt("1234567890123456789012345678901234567890"); + +const bigintFromNumber = BigInt(10); // same as 10n +``` + +## Math operators + +`BigInt` can mostly be used like a regular number, for example: + +```js run +alert(1n + 2n); // 3 + +alert(5n / 2n); // 2 +``` + +Please note: the division `5/2` returns the result rounded towards zero, without the decimal part. All operations on bigints return bigints. + +We can't mix bigints and regular numbers: + +```js run +alert(1n + 2); // Error: Cannot mix BigInt and other types +``` + +We should explicitly convert them if needed: using either `BigInt()` or `Number()`, like this: + +```js run +let bigint = 1n; +let number = 2; + +// number to bigint +alert(bigint + BigInt(number)); // 3 + +// bigint to number +alert(Number(bigint) + number); // 3 +``` + +The conversion operations are always silent, never give errors, but if the bigint is too huge and won't fit the number type, then extra bits will be cut off, so we should be careful doing such conversion. + +````smart header="The unary plus is not supported on bigints" +The unary plus operator `+value` is a well-known way to convert `value` to a number. + +In order to avoid confusion, it's not supported on bigints: +```js run +let bigint = 1n; + +alert( +bigint ); // error +``` +So we should use `Number()` to convert a bigint to a number. +```` + +## Comparisons + +Comparisons, such as `<`, `>` work with bigints and numbers just fine: + +```js run +alert( 2n > 1n ); // true + +alert( 2n > 1 ); // true +``` + +Please note though, as numbers and bigints belong to different types, they can be equal `==`, but not strictly equal `===`: + +```js run +alert( 1 == 1n ); // true + +alert( 1 === 1n ); // false +``` + +## Boolean operations + +When inside `if` or other boolean operations, bigints behave like numbers. + +For instance, in `if`, bigint `0n` is falsy, other values are truthy: + +```js run +if (0n) { + // never executes +} +``` + +Boolean operators, such as `||`, `&&` and others also work with bigints similar to numbers: + +```js run +alert( 1n || 2 ); // 1 (1n is considered truthy) + +alert( 0n || 2 ); // 2 (0n is considered falsy) +``` + +## Polyfills + +Polyfilling bigints is tricky. The reason is that many JavaScript operators, such as `+`, `-` and so on behave differently with bigints compared to regular numbers. + +For example, division of bigints always returns a bigint (rounded if necessary). + +To emulate such behavior, a polyfill would need to analyze the code and replace all such operators with its functions. But doing so is cumbersome and would cost a lot of performance. + +So, there's no well-known good polyfill. + +Although, the other way around is proposed by the developers of [JSBI](https://github.com/GoogleChromeLabs/jsbi) library. + +This library implements big numbers using its own methods. We can use them instead of native bigints: + +| Operation | native `BigInt` | JSBI | +|-----------|-----------------|------| +| Creation from Number | `a = BigInt(789)` | `a = JSBI.BigInt(789)` | +| Addition | `c = a + b` | `c = JSBI.add(a, b)` | +| Subtraction | `c = a - b` | `c = JSBI.subtract(a, b)` | +| ... | ... | ... | + +...And then use the polyfill (Babel plugin) to convert JSBI calls to native bigints for those browsers that support them. + +In other words, this approach suggests that we write code in JSBI instead of native bigints. But JSBI works with numbers as with bigints internally, emulates them closely following the specification, so the code will be "bigint-ready". + +We can use such JSBI code "as is" for engines that don't support bigints and for those that do support - the polyfill will convert the calls to native bigints. + +## References + +- [MDN docs on BigInt](mdn:/JavaScript/Reference/Global_Objects/BigInt). +- [Specification](https://tc39.es/ecma262/#sec-bigint-objects). diff --git a/1-js/99-js-misc/06-unicode/article.md b/1-js/99-js-misc/06-unicode/article.md new file mode 100644 index 000000000..4f144f824 --- /dev/null +++ b/1-js/99-js-misc/06-unicode/article.md @@ -0,0 +1,172 @@ + +# Unicode, String internals + +```warn header="Advanced knowledge" +The section goes deeper into string internals. This knowledge will be useful for you if you plan to deal with emoji, rare mathematical or hieroglyphic characters, or other rare symbols. +``` + +As we already know, JavaScript strings are based on [Unicode](https://en.wikipedia.org/wiki/Unicode): each character is represented by a byte sequence of 1-4 bytes. + +JavaScript allows us to insert a character into a string by specifying its hexadecimal Unicode code with one of these three notations: + +- `\xXX` + + `XX` must be two hexadecimal digits with a value between `00` and `FF`, then `\xXX` is the character whose Unicode code is `XX`. + + Because the `\xXX` notation supports only two hexadecimal digits, it can be used only for the first 256 Unicode characters. + + These first 256 characters include the Latin alphabet, most basic syntax characters, and some others. For example, `"\x7A"` is the same as `"z"` (Unicode `U+007A`). + + ```js run + alert( "\x7A" ); // z + alert( "\xA9" ); // ©, the copyright symbol + ``` + +- `\uXXXX` + `XXXX` must be exactly 4 hex digits with the value between `0000` and `FFFF`, then `\uXXXX` is the character whose Unicode code is `XXXX`. + + Characters with Unicode values greater than `U+FFFF` can also be represented with this notation, but in this case, we will need to use a so called surrogate pair (we will talk about surrogate pairs later in this chapter). + + ```js run + alert( "\u00A9" ); // ©, the same as \xA9, using the 4-digit hex notation + alert( "\u044F" ); // я, the Cyrillic alphabet letter + alert( "\u2191" ); // ↑, the arrow up symbol + ``` + +- `\u{X…XXXXXX}` + + `X…XXXXXX` must be a hexadecimal value of 1 to 6 bytes between `0` and `10FFFF` (the highest code point defined by Unicode). This notation allows us to easily represent all existing Unicode characters. + + ```js run + alert( "\u{20331}" ); // 佫, a rare Chinese character (long Unicode) + alert( "\u{1F60D}" ); // 😍, a smiling face symbol (another long Unicode) + ``` + +## Surrogate pairs + +All frequently used characters have 2-byte codes (4 hex digits). Letters in most European languages, numbers, and the basic unified CJK ideographic sets (CJK -- from Chinese, Japanese, and Korean writing systems), have a 2-byte representation. + +Initially, JavaScript was based on UTF-16 encoding that only allowed 2 bytes per character. But 2 bytes only allow 65536 combinations and that's not enough for every possible symbol of Unicode. + +So rare symbols that require more than 2 bytes are encoded with a pair of 2-byte characters called "a surrogate pair". + +As a side effect, the length of such symbols is `2`: + +```js run +alert( '𝒳'.length ); // 2, MATHEMATICAL SCRIPT CAPITAL X +alert( '😂'.length ); // 2, FACE WITH TEARS OF JOY +alert( '𩷶'.length ); // 2, a rare Chinese character +``` + +That's because surrogate pairs did not exist at the time when JavaScript was created, and thus are not correctly processed by the language! + +We actually have a single symbol in each of the strings above, but the `length` property shows a length of `2`. + +Getting a symbol can also be tricky, because most language features treat surrogate pairs as two characters. + +For example, here we can see two odd characters in the output: + +```js run +alert( '𝒳'[0] ); // shows strange symbols... +alert( '𝒳'[1] ); // ...pieces of the surrogate pair +``` + +Pieces of a surrogate pair have no meaning without each other. So the alerts in the example above actually display garbage. + +Technically, surrogate pairs are also detectable by their codes: if a character has the code in the interval of `0xd800..0xdbff`, then it is the first part of the surrogate pair. The next character (second part) must have the code in interval `0xdc00..0xdfff`. These intervals are reserved exclusively for surrogate pairs by the standard. + +So the methods [String.fromCodePoint](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/fromCodePoint) and [str.codePointAt](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/codePointAt) were added in JavaScript to deal with surrogate pairs. + +They are essentially the same as [String.fromCharCode](mdn:js/String/fromCharCode) and [str.charCodeAt](mdn:js/String/charCodeAt), but they treat surrogate pairs correctly. + +One can see the difference here: + +```js run +// charCodeAt is not surrogate-pair aware, so it gives codes for the 1st part of 𝒳: + +alert( '𝒳'.charCodeAt(0).toString(16) ); // d835 + +// codePointAt is surrogate-pair aware +alert( '𝒳'.codePointAt(0).toString(16) ); // 1d4b3, reads both parts of the surrogate pair +``` + +That said, if we take from position 1 (and that's rather incorrect here), then they both return only the 2nd part of the pair: + +```js run +alert( '𝒳'.charCodeAt(1).toString(16) ); // dcb3 +alert( '𝒳'.codePointAt(1).toString(16) ); // dcb3 +// meaningless 2nd half of the pair +``` + +You will find more ways to deal with surrogate pairs later in the chapter <info:iterable>. There are probably special libraries for that too, but nothing famous enough to suggest here. + +````warn header="Takeaway: splitting strings at an arbitrary point is dangerous" +We can't just split a string at an arbitrary position, e.g. take `str.slice(0, 4)` and expect it to be a valid string, e.g.: + +```js run +alert( 'hi 😂'.slice(0, 4) ); // hi [?] +``` + +Here we can see a garbage character (first half of the smile surrogate pair) in the output. + +Just be aware of it if you intend to reliably work with surrogate pairs. May not be a big problem, but at least you should understand what happens. +```` + +## Diacritical marks and normalization + +In many languages, there are symbols that are composed of the base character with a mark above/under it. + +For instance, the letter `a` can be the base character for these characters: `àáâäãåā`. + +Most common "composite" characters have their own code in the Unicode table. But not all of them, because there are too many possible combinations. + +To support arbitrary compositions, the Unicode standard allows us to use several Unicode characters: the base character followed by one or many "mark" characters that "decorate" it. + +For instance, if we have `S` followed by the special "dot above" character (code `\u0307`), it is shown as Ṡ. + +```js run +alert( 'S\u0307' ); // Ṡ +``` + +If we need an additional mark above the letter (or below it) -- no problem, just add the necessary mark character. + +For instance, if we append a character "dot below" (code `\u0323`), then we'll have "S with dots above and below": `Ṩ`. + +For example: + +```js run +alert( 'S\u0307\u0323' ); // Ṩ +``` + +This provides great flexibility, but also an interesting problem: two characters may visually look the same, but be represented with different Unicode compositions. + +For instance: + +```js run +let s1 = 'S\u0307\u0323'; // Ṩ, S + dot above + dot below +let s2 = 'S\u0323\u0307'; // Ṩ, S + dot below + dot above + +alert( `s1: ${s1}, s2: ${s2}` ); + +alert( s1 == s2 ); // false though the characters look identical (?!) +``` + +To solve this, there exists a "Unicode normalization" algorithm that brings each string to the single "normal" form. + +It is implemented by [str.normalize()](mdn:js/String/normalize). + +```js run +alert( "S\u0307\u0323".normalize() == "S\u0323\u0307".normalize() ); // true +``` + +It's funny that in our situation `normalize()` actually brings together a sequence of 3 characters to one: `\u1e68` (S with two dots). + +```js run +alert( "S\u0307\u0323".normalize().length ); // 1 + +alert( "S\u0307\u0323".normalize() == "\u1e68" ); // true +``` + +In reality, this is not always the case. The reason is that the symbol `Ṩ` is "common enough", so Unicode creators included it in the main table and gave it the code. + +If you want to learn more about normalization rules and variants -- they are described in the appendix of the Unicode standard: [Unicode Normalization Forms](https://www.unicode.org/reports/tr15/), but for most practical purposes the information from this section is enough. diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/article.md b/1-js/99-js-misc/07-weakref-finalizationregistry/article.md new file mode 100644 index 000000000..777bf703c --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/article.md @@ -0,0 +1,483 @@ + +# WeakRef and FinalizationRegistry + +```warn header="\"Hidden\" features of the language" +This article covers a very narrowly focused topic, that most developers extremely rarely encounter in practice (and may not even be aware of its existence). + +We recommend skipping this chapter if you have just started learning JavaScript. +``` + +Recalling the basic concept of the *reachability principle* from the <info:garbage-collection> chapter, +we can note that the JavaScript engine is guaranteed to keep values in memory that are accessible or in use. + +For example: + + +```js +// the user variable holds a strong reference to the object +let user = { name: "John" }; + +// let's overwrite the value of the user variable +user = null; + +// the reference is lost and the object will be deleted from memory + +``` + +Or a similar, but slightly more complicated code with two strong references: + +```js +// the user variable holds a strong reference to the object +let user = { name: "John" }; + +// copied the strong reference to the object into the admin variable +*!* +let admin = user; +*/!* + +// let's overwrite the value of the user variable +user = null; + +// the object is still reachable through the admin variable +``` +The object `{ name: "John" }` would only be deleted from memory if there were no strong references to it (if we also overwrote the value of the `admin` variable). + +In JavaScript, there is a concept called `WeakRef`, which behaves slightly differently in this case. + + +````smart header="Terms: \"Strong reference\", \"Weak reference\"" +**Strong reference** - is a reference to an object or value, that prevents them from being deleted by the garbage collector. Thereby, keeping the object or value in memory, to which it points. + +This means, that the object or value remains in memory and is not collected by the garbage collector as long, as there are active strong references to it. + +In JavaScript, ordinary references to objects are strong references. For example: + +```js +// the user variable holds a strong reference to this object +let user = { name: "John" }; +``` +**Weak reference** - is a reference to an object or value, that does *not* prevent them from being deleted by the garbage collector. +An object or value can be deleted by the garbage collector if, the only remaining references to them are weak references. +```` + +## WeakRef + + +````warn header="Note of caution" +Before we dive into it, it is worth noting that the correct use of the structures discussed in this article requires very careful thought, and they are best avoided if possible. +```` + +`WeakRef` - is an object, that contains a weak reference to another object, called `target` or `referent`. + +The peculiarity of `WeakRef` is that it does not prevent the garbage collector from deleting its referent-object. In other words, a `WeakRef` object does not keep the `referent` object alive. + +Now let's take the `user` variable as the "referent" and create a weak reference from it to the `admin` variable. +To create a weak reference, you need to use the `WeakRef` constructor, passing in the target object (the object you want a weak reference to). + +In our case — this is the `user` variable: + + +```js +// the user variable holds a strong reference to the object +let user = { name: "John" }; + +// the admin variable holds a weak reference to the object +*!* +let admin = new WeakRef(user); +*/!* + +``` + +The diagram below depicts two types of references: a strong reference using the `user` variable and a weak reference using the `admin` variable: + + + +Then, at some point, we stop using the `user` variable - it gets overwritten, goes out of scope, etc., while keeping the `WeakRef` instance in the `admin` variable: + +```js +// let's overwrite the value of the user variable +user = null; +``` + +A weak reference to an object is not enough to keep it "alive". When the only remaining references to a referent-object are weak references, the garbage collector is free to destroy this object and use its memory for something else. + +However, until the object is actually destroyed, the weak reference may return it, even if there are no more strong references to this object. +That is, our object becomes a kind of "[Schrödinger's cat](https://en.wikipedia.org/wiki/Schr%C3%B6dinger%27s_cat)" – we cannot know for sure whether it's "alive" or "dead": + + + +At this point, to get the object from the `WeakRef` instance, we will use its `deref()` method. + +The `deref()` method returns the referent-object that the `WeakRef` points to, if the object is still in memory. If the object has been deleted by the garbage collector, then the `deref()` method will return `undefined`: + + +```js +let ref = admin.deref(); + +if (ref) { + // the object is still accessible: we can perform any manipulations with it +} else { + // the object has been collected by the garbage collector +} +``` + +## WeakRef use cases + +`WeakRef` is typically used to create caches or [associative arrays](https://en.wikipedia.org/wiki/Associative_array) that store resource-intensive objects. +This allows one to avoid preventing these objects from being collected by the garbage collector solely based on their presence in the cache or associative array. + +One of the primary examples - is a situation when we have numerous binary image objects (for instance, represented as `ArrayBuffer` or `Blob`), and we want to associate a name or path with each image. +Existing data structures are not quite suitable for these purposes: + +- Using `Map` to create associations between names and images, or vice versa, will keep the image objects in memory since they are present in the `Map` as keys or values. +- `WeakMap` is ineligible for this goal either: because the objects represented as `WeakMap` keys use weak references, and are not protected from deletion by the garbage collector. + +But, in this situation, we need a data structure that would use weak references in its values. + +For this purpose, we can use a `Map` collection, whose values are `WeakRef` instances referring to the large objects we need. +Consequently, we will not keep these large and unnecessary objects in memory longer than they should be. + +Otherwise, this is a way to get the image object from the cache if it is still reachable. +If it has been garbage collected, we will re-generate or re-download it again. + +This way, less memory is used in some situations. + +## Example №1: using WeakRef for caching + +Below is a code snippet that demonstrates the technique of using `WeakRef`. + +In short, we use a `Map` with string keys and `WeakRef` objects as their values. +If the `WeakRef` object has not been collected by the garbage collector, we get it from the cache. +Otherwise, we re-download it again and put it in the cache for further possible reuse: + +```js +function fetchImg() { + // abstract function for downloading images... +} + +function weakRefCache(fetchImg) { // (1) + const imgCache = new Map(); // (2) + + return (imgName) => { // (3) + const cachedImg = imgCache.get(imgName); // (4) + + if (cachedImg?.deref()) { // (5) + return cachedImg?.deref(); + } + + const newImg = fetchImg(imgName); // (6) + imgCache.set(imgName, new WeakRef(newImg)); // (7) + + return newImg; + }; +} + +const getCachedImg = weakRefCache(fetchImg); +``` + +Let's delve into the details of what happened here: +1. `weakRefCache` - is a higher-order function that takes another function, `fetchImg`, as an argument. In this example, we can neglect a detailed description of the `fetchImg` function, since it can be any logic for downloading images. +2. `imgCache` - is a cache of images, that stores cached results of the `fetchImg` function, in the form of string keys (image name) and `WeakRef` objects as their values. +3. Return an anonymous function that takes the image name as an argument. This argument will be used as a key for the cached image. +4. Trying to get the cached result from the cache, using the provided key (image name). +5. If the cache contains a value for the specified key, and the `WeakRef` object has not been deleted by the garbage collector, return the cached result. +6. If there is no entry in the cache with the requested key, or `deref()` method returns `undefined` (meaning that the `WeakRef` object has been garbage collected), the `fetchImg` function downloads the image again. +7. Put the downloaded image into the cache as a `WeakRef` object. + +Now we have a `Map` collection, where the keys - are image names as strings, and values - are `WeakRef` objects containing the images themselves. + +This technique helps to avoid allocating a large amount of memory for resource-intensive objects, that nobody uses anymore. +It also saves memory and time in case of reusing cached objects. + +Here is a visual representation of what this code looks like: + + + +But, this implementation has its drawbacks: over time, `Map` will be filled with strings as keys, that point to a `WeakRef`, whose referent-object has already been garbage collected: + + + +One way to handle this problem - is to periodically scavenge the cache and clear out "dead" entries. +Another way - is to use finalizers, which we will explore next. + + +## Example №2: Using WeakRef to track DOM objects + +Another use case for `WeakRef` - is tracking DOM objects. + +Let's imagine a scenario where some third-party code or library interacts with elements on our page as long as they exist in the DOM. +For example, it could be an external utility for monitoring and notifying about the system's state (commonly so-called "logger" – a program that sends informational messages called "logs"). + +Interactive example: + +[codetabs height=420 src="weakref-dom"] + +When the "Start sending messages" button is clicked, in the so-called "logs display window" (an element with the `.window__body` class), messages (logs) start to appear. + +But, as soon as this element is deleted from the DOM, the logger should stop sending messages. +To reproduce the removal of this element, just click the "Close" button in the top right corner. + +In order not to complicate our work, and not to notify third-party code every time our DOM-element is available, and when it is not, it will be enough to create a weak reference to it using `WeakRef`. + +Once the element is removed from the DOM, the logger will notice it and stop sending messages. + +Now let's take a closer look at the source code (*tab `index.js`*): + +1. Get the DOM-element of the "Start sending messages" button. +2. Get the DOM-element of the "Close" button. +3. Get the DOM-element of the logs display window using the `new WeakRef()` constructor. This way, the `windowElementRef` variable holds a weak reference to the DOM-element. +4. Add an event listener on the "Start sending messages" button, responsible for starting the logger when clicked. +5. Add an event listener on the "Close" button, responsible for closing the logs display window when clicked. +6. Use `setInterval` to start displaying a new message every second. +7. If the DOM-element of the logs display window is still accessible and kept in memory, create and send a new message. +8. If the `deref()` method returns `undefined`, it means that the DOM-element has been deleted from memory. In this case, the logger stops displaying messages and clears the timer. +9. `alert`, which will be called, after the DOM-element of the logs display window is deleted from memory (i.e. after clicking the "Close" button). **Note, that deletion from memory may not happen immediately, as it depends only on the internal mechanisms of the garbage collector.** + + We cannot control this process directly from the code. However, despite this, we still have the option to force garbage collection from the browser. + + In Google Chrome, for example, to do this, you need to open the developer tools (`key:Ctrl` + `key:Shift` + `key:J` on Windows/Linux or `key:Option` + `key:⌘` + `key:J` on macOS), go to the "Performance" tab, and click on the bin icon button – "Collect garbage": + +  + + <br> + This functionality is supported in most modern browsers. After the actions are taken, the <code>alert</code> will trigger immediately. + +## FinalizationRegistry + +Now it is time to talk about finalizers. Before we move on, let's clarify the terminology: + +**Cleanup callback (finalizer)** - is a function that is executed, when an object, registered in the `FinalizationRegistry`, is deleted from memory by the garbage collector. + +Its purpose - is to provide the ability to perform additional operations, related to the object, after it has been finally deleted from memory. + +**Registry** (or `FinalizationRegistry`) - is a special object in JavaScript that manages the registration and unregistration of objects and their cleanup callbacks. + +This mechanism allows registering an object to track and associate a cleanup callback with it. +Essentially it is a structure that stores information about registered objects and their cleanup callbacks, and then automatically invokes those callbacks when the objects are deleted from memory. + +To create an instance of the `FinalizationRegistry`, it needs to call its constructor, which takes a single argument - the cleanup callback (finalizer). + +Syntax: + +```js +function cleanupCallback(heldValue) { + // cleanup callback code +} + +const registry = new FinalizationRegistry(cleanupCallback); +``` + +Here: + +- `cleanupCallback` - a cleanup callback that will be automatically called when a registered object is deleted from memory. +- `heldValue` - the value that is passed as an argument to the cleanup callback. If `heldValue` is an object, the registry keeps a strong reference to it. +- `registry` - an instance of `FinalizationRegistry`. + +`FinalizationRegistry` methods: + +- `register(target, heldValue [, unregisterToken])` - used to register objects in the registry. + + `target` - the object being registered for tracking. If the `target` is garbage collected, the cleanup callback will be called with `heldValue` as its argument. + + Optional `unregisterToken` – an unregistration token. It can be passed to unregister an object before the garbage collector deletes it. Typically, the `target` object is used as `unregisterToken`, which is the standard practice. +- `unregister(unregisterToken)` - the `unregister` method is used to unregister an object from the registry. It takes one argument - `unregisterToken` (the unregister token that was obtained when registering the object). + +Now let's move on to a simple example. Let's use the already-known `user` object and create an instance of `FinalizationRegistry`: + +```js +let user = { name: "John" }; + +const registry = new FinalizationRegistry((heldValue) => { + console.log(`${heldValue} has been collected by the garbage collector.`); +}); +``` + +Then, we will register the object, that requires a cleanup callback by calling the `register` method: + +```js +registry.register(user, user.name); +``` + +The registry does not keep a strong reference to the object being registered, as this would defeat its purpose. If the registry kept a strong reference, then the object would never be garbage collected. + +If the object is deleted by the garbage collector, our cleanup callback may be called at some point in the future, with the `heldValue` passed to it: + +```js +// When the user object is deleted by the garbage collector, the following message will be printed in the console: +"John has been collected by the garbage collector." +``` + +There are also situations where, even in implementations that use a cleanup callback, there is a chance that it will not be called. + +For example: +- When the program fully terminates its operation (for example, when closing a tab in a browser). +- When the `FinalizationRegistry` instance itself is no longer reachable to JavaScript code. + If the object that creates the `FinalizationRegistry` instance goes out of scope or is deleted, the cleanup callbacks registered in that registry might also not be invoked. + +## Caching with FinalizationRegistry + +Returning to our *weak* cache example, we can notice the following: +- Even though the values wrapped in the `WeakRef` have been collected by the garbage collector, there is still an issue of "memory leakage" in the form of the remaining keys, whose values have been collected by the garbage collector. + +Here is an improved caching example using `FinalizationRegistry`: + +```js +function fetchImg() { + // abstract function for downloading images... +} + +function weakRefCache(fetchImg) { + const imgCache = new Map(); + + *!* + const registry = new FinalizationRegistry((imgName) => { // (1) + const cachedImg = imgCache.get(imgName); + if (cachedImg && !cachedImg.deref()) imgCache.delete(imgName); + }); + */!* + + return (imgName) => { + const cachedImg = imgCache.get(imgName); + + if (cachedImg?.deref()) { + return cachedImg?.deref(); + } + + const newImg = fetchImg(imgName); + imgCache.set(imgName, new WeakRef(newImg)); + *!* + registry.register(newImg, imgName); // (2) + */!* + + return newImg; + }; +} + +const getCachedImg = weakRefCache(fetchImg); +``` + +1. To manage the cleanup of "dead" cache entries, when the associated `WeakRef` objects are collected by the garbage collector, we create a `FinalizationRegistry` cleanup registry. + + The important point here is, that in the cleanup callback, it should be checked, if the entry was deleted by the garbage collector and not re-added, in order not to delete a "live" entry. +2. Once the new value (image) is downloaded and put into the cache, we register it in the finalizer registry to track the `WeakRef` object. + +This implementation contains only actual or "live" key/value pairs. +In this case, each `WeakRef` object is registered in the `FinalizationRegistry`. +And after the objects are cleaned up by the garbage collector, the cleanup callback will delete all `undefined` values. + +Here is a visual representation of the updated code: + + + +A key aspect of the updated implementation is that finalizers allow parallel processes to be created between the "main" program and cleanup callbacks. +In the context of JavaScript, the "main" program - is our JavaScript-code, that runs and executes in our application or web page. + +Hence, from the moment an object is marked for deletion by the garbage collector, and to the actual execution of the cleanup callback, there may be a certain time gap. +It is important to understand that during this time gap, the main program can make any changes to the object or even bring it back to memory. + +That's why, in the cleanup callback, we must check to see if an entry has been added back to the cache by the main program to avoid deleting "live" entries. +Similarly, when searching for a key in the cache, there is a chance that the value has been deleted by the garbage collector, but the cleanup callback has not been executed yet. + +Such situations require special attention if you are working with `FinalizationRegistry`. + +## Using WeakRef and FinalizationRegistry in practice + +Moving from theory to practice, imagine a real-life scenario, where a user synchronizes their photos on a mobile device with some cloud service +(such as [iCloud](https://en.wikipedia.org/wiki/ICloud) or [Google Photos](https://en.wikipedia.org/wiki/Google_Photos)), +and wants to view them from other devices. In addition to the basic functionality of viewing photos, such services offer a lot of additional features, for example: + +- Photo editing and video effects. +- Creating "memories" and albums. +- Video montage from a series of photos. +- ...and much more. + +Here, as an example, we will use a fairly primitive implementation of such a service. +The main point - is to show a possible scenario of using `WeakRef` and `FinalizationRegistry` together in real life. + +Here is what it looks like: + + + +<br> +On the left side, there is a cloud library of photos (they are displayed as thumbnails). +We can select the images we need and create a collage, by clicking the "Create collage" button on the right side of the page. +Then, the resulting collage can be downloaded as an image. +</br><br> + +To increase page loading speed, it would be reasonable to download and display photo thumbnails in *compressed* quality. +But, to create a collage from selected photos, download and use them in *full-size* quality. + +Below, we can see, that the intrinsic size of the thumbnails is 240x240 pixels. +The size was chosen on purpose to increase loading speed. +Moreover, we do not need full-size photos in preview mode. + + + +<br> +Let's assume, that we need to create a collage of 4 photos: we select them, and then click the "Create collage" button. +At this stage, the already known to us <code>weakRefCache</code> function checks whether the required image is in the cache. +If not, it downloads it from the cloud and puts it in the cache for further use. +This happens for each selected image: +</br><br> + + + +</br> + +Paying attention to the output in the console, you can see, which of the photos were downloaded from the cloud - this is indicated by <span style="background-color:#133159;color:white;font-weight:500">FETCHED_IMAGE</span>. +Since this is the first attempt to create a collage, this means, that at this stage the "weak cache" was still empty, and all the photos were downloaded from the cloud and put in it. + +But, along with the process of downloading images, there is also a process of memory cleanup by the garbage collector. +This means, that the object stored in the cache, which we refer to, using a weak reference, is deleted by the garbage collector. +And our finalizer executes successfully, thereby deleting the key, by which the image was stored in the cache. +<span style="background-color:#901e30;color:white;font-weight:500;">CLEANED_IMAGE</span> notifies us about it: + + + +<br> +Next, we realize that we do not like the resulting collage, and decide to change one of the images and create a new one. +To do this, just deselect the unnecessary image, select another one, and click the "Create collage" button again: +</br><br> + + + +<br> +But this time not all images were downloaded from the network, and one of them was taken from the weak cache: the <span style="background-color:#385950;color:white;font-weight:500;">CACHED_IMAGE</span> message tells us about it. +This means that at the time of collage creation, the garbage collector had not yet deleted our image, and we boldly took it from the cache, +thereby reducing the number of network requests and speeding up the overall time of the collage creation process: +</br><br> + + + +<br> +Let's "play around" a little more, by replacing one of the images again and creating a new collage: +</br><br> + + + +<br> +This time the result is even more impressive. Of the 4 images selected, 3 of them were taken from the weak cache, and only one had to be downloaded from the network. +The reduction in network load was about 75%. Impressive, isn't it? +</br><br> + + + +</br> + +Of course, it is important to remember, that such behavior is not guaranteed, and depends on the specific implementation and operation of the garbage collector. + +Based on this, a completely logical question immediately arises: why do not we use an ordinary cache, where we can manage its entities ourselves, instead of relying on the garbage collector? +That's right, in the vast majority of cases there is no need to use `WeakRef` and `FinalizationRegistry`. + +Here, we simply demonstrated an alternative implementation of similar functionality, using a non-trivial approach with interesting language features. +Still, we cannot rely on this example, if we need a constant and predictable result. + +You can [open this example in the sandbox](sandbox:weakref-finalizationregistry). + +## Summary + +`WeakRef` - designed to create weak references to objects, allowing them to be deleted from memory by the garbage collector if there are no longer strong references to them. +This is beneficial for addressing excessive memory usage and optimizing the utilization of system resources in applications. + +`FinalizationRegistry` - is a tool for registering callbacks, that are executed when objects that are no longer strongly referenced, are destroyed. +This allows releasing resources associated with the object or performing other necessary operations before deleting the object from memory. \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/google-chrome-developer-tools.png b/1-js/99-js-misc/07-weakref-finalizationregistry/google-chrome-developer-tools.png new file mode 100644 index 000000000..021637342 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/google-chrome-developer-tools.png differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.css b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.css new file mode 100644 index 000000000..f6df812d0 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.css @@ -0,0 +1,49 @@ +.app { + display: flex; + flex-direction: column; + gap: 16px; +} + +.start-messages { + width: fit-content; +} + +.window { + width: 100%; + border: 2px solid #464154; + overflow: hidden; +} + +.window__header { + position: sticky; + padding: 8px; + display: flex; + justify-content: space-between; + align-items: center; + background-color: #736e7e; +} + +.window__title { + margin: 0; + font-size: 24px; + font-weight: 700; + color: white; + letter-spacing: 1px; +} + +.window__button { + padding: 4px; + background: #4f495c; + outline: none; + border: 2px solid #464154; + color: white; + font-size: 16px; + cursor: pointer; +} + +.window__body { + height: 250px; + padding: 16px; + overflow: scroll; + background-color: #736e7e33; +} \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.html b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.html new file mode 100644 index 000000000..7f93af4c7 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.html @@ -0,0 +1,28 @@ +<!DOCTYPE HTML> +<html lang="en"> + +<head> + <meta charset="utf-8"> + <link rel="stylesheet" href="index.css"> + <title>WeakRef DOM Logger</title> +</head> + +<body> + +<div class="app"> + <button class="start-messages">Start sending messages</button> + <div class="window"> + <div class="window__header"> + <p class="window__title">Messages:</p> + <button class="window__button">Close</button> + </div> + <div class="window__body"> + No messages. + </div> + </div> +</div> + + +<script type="module" src="index.js"></script> +</body> +</html> diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.js b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.js new file mode 100644 index 000000000..ea55b4478 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-dom.view/index.js @@ -0,0 +1,24 @@ +const startMessagesBtn = document.querySelector('.start-messages'); // (1) +const closeWindowBtn = document.querySelector('.window__button'); // (2) +const windowElementRef = new WeakRef(document.querySelector(".window__body")); // (3) + +startMessagesBtn.addEventListener('click', () => { // (4) + startMessages(windowElementRef); + startMessagesBtn.disabled = true; +}); + +closeWindowBtn.addEventListener('click', () => document.querySelector(".window__body").remove()); // (5) + + +const startMessages = (element) => { + const timerId = setInterval(() => { // (6) + if (element.deref()) { // (7) + const payload = document.createElement("p"); + payload.textContent = `Message: System status OK: ${new Date().toLocaleTimeString()}`; + element.deref().append(payload); + } else { // (8) + alert("The element has been deleted."); // (9) + clearInterval(timerId); + } + }, 1000); +}; \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-01.svg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-01.svg new file mode 100644 index 000000000..2a507dbcd --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-01.svg @@ -0,0 +1,32 @@ +<?xml version="1.0" encoding="utf-8"?> +<svg viewBox="-38.324 -109.673 191.121 281.642" width="191.121px" height="281.642px" xmlns="http://www.w3.org/2000/svg" xmlns:bx="https://boxy-svg.com"> + <defs> + <style bx:fonts="Open Sans">@import url(https://fonts.googleapis.com/css2?family=Open+Sans%3Aital%2Cwght%400%2C300%3B0%2C400%3B0%2C500%3B0%2C600%3B0%2C700%3B0%2C800%3B1%2C300%3B1%2C400%3B1%2C500%3B1%2C600%3B1%2C700%3B1%2C800&display=swap);</style> + </defs> + <g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1" transform="matrix(1.6492669582366946, 0, 0, 1.6492669582366946, -81.42222595214844, 62.8305015563965)" style=""> + <g id="memory-user-john-admin.svg" transform="matrix(1, 0, 0, 1, -1.759865, 4.803441)"> + <text id="user" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; font-weight: 600; white-space: pre;" transform="matrix(0.735667, 0, 0, 0.753868, 7.56361, 10.722544)" x="31.917" y="-40.191">user</text> + <path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 39.02 4.206 L 125.8 4.206 L 125.8 48.227 L 39.02 48.227 L 39.02 4.206 Z" style=""/> + <text id="name:-"John"" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 9px; font-weight: 500; white-space: pre;" x="54.086" y="34.58">name: "John"</text> + <text id="Object" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 10.5px; font-weight: 500; white-space: pre;" x="64.679" y="23.487">Object</text> + <path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 36.398 -102.608 L 129.398 -102.608 L 129.398 -76.608 L 36.398 -76.608 L 36.398 -102.608 Z"/> + <text id="<global>" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 10.6px; font-weight: 500; white-space: pre;" x="59.233" y="-86.001"> <global></text> + <path d="M 105.999 1.57 L 113.013 13.895 L 98.985 13.895 L 105.999 1.57 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 98.985 1.57 14.028 12.325 0.5 0 1@df7d9fa9"/> + <g> + <rect x="105.024" y="-69.436" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1.024052, -0.03572, 31.45084)" style=""> + <rect x="105.024" y="-69.051" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1, -9.930843, 12.564639)"> + <path d="M 66.874 14.118 L 73.888 26.443 L 59.86 26.443 L 66.874 14.118 Z" style="fill-rule: nonzero; fill: rgb(192, 99, 52);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 59.86 14.118 14.028 12.325 0.5 0 1@9d4d6098"/> + <rect x="65.833" y="-81.984" width="2.056" height="55.884" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(192, 99, 52);"/> + </g> + <text id="text-1" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 14px; font-weight: 600; white-space: pre;" transform="matrix(0.735667, 0, 0, 0.753868, 87.03299, 10.790039)" x="31.917" y="-40.191">admin</text> + </g> + </g> +</svg> \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-02.svg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-02.svg new file mode 100644 index 000000000..6cc199a12 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-02.svg @@ -0,0 +1,33 @@ +<?xml version="1.0" encoding="utf-8"?> +<svg viewBox="-56.888 -212.944 192.167 294.011" width="178.044px" height="272.403px" xmlns="http://www.w3.org/2000/svg" xmlns:bx="https://boxy-svg.com"> + <defs> + <style bx:fonts="Open Sans">@import url(https://fonts.googleapis.com/css2?family=Open+Sans%3Aital%2Cwght%400%2C300%3B0%2C400%3B0%2C500%3B0%2C600%3B0%2C700%3B0%2C800%3B1%2C300%3B1%2C400%3B1%2C500%3B1%2C600%3B1%2C700%3B1%2C800&display=swap);</style> + </defs> + <g id="garbage-collection" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1" transform="matrix(1.708757996559143, 0, 0, 1.708757996559143, -109.71231842041016, -35.19435119628906)" style=""> + <g id="memory-user-john-admin.svg" transform="matrix(1, 0, 0, 1, -1.759865, 4.803441)"> + <path id="Rectangle-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 39.02 4.206 L 125.8 4.206 L 125.8 48.227 L 39.02 48.227 L 39.02 4.206 Z" style=""/> + <path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 36.398 -102.608 L 129.398 -102.608 L 129.398 -76.608 L 36.398 -76.608 L 36.398 -102.608 Z"/> + <text id="<global>" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 10.8px; font-weight: 500; white-space: pre;" x="57.977" y="-85.926"> <global></text> + <path d="M 105.999 1.57 L 113.013 13.895 L 98.985 13.895 L 105.999 1.57 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 98.985 1.57 14.028 12.325 0.5 0 1@df7d9fa9"/> + <g style="" transform="matrix(0.7989, 0, 0, 0.7989, 13.076554, 6.083323)"> + <text id="name:-"John"" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 12px; font-weight: 500; white-space: pre;" x="48.212" y="37.89">name: "John"</text> + <text id="Object" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; font-weight: 500; white-space: pre;" x="61.598" y="23.794">Object</text> + </g> + <g> + <rect x="105.024" y="-69.436" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1.024052, -0.03572, 31.45084)" style=""> + <rect x="105.024" y="-69.051" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <text id="text-1" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 14px; font-weight: 600; white-space: pre;" transform="matrix(0.735667, 0, 0, 0.753868, 86.426659, 10.790039)" x="31.917" y="-40.191">admin</text> + </g> + </g> + <g transform="matrix(0.529145, 0, 0, 0.529145, -21.841589, -83.135681)" style=""> + <path d="M80.179,13.758c-18.342-18.342-48.08-18.342-66.422,0c-18.342,18.341-18.342,48.08,0,66.421 c18.342,18.342,48.08,18.342,66.422,0C98.521,61.837,98.521,32.099,80.179,13.758z M44.144,83.117 c-4.057,0-7.001-3.071-7.001-7.305c0-4.291,2.987-7.404,7.102-7.404c4.123,0,7.001,3.044,7.001,7.404 C51.246,80.113,48.326,83.117,44.144,83.117z M54.73,44.921c-4.15,4.905-5.796,9.117-5.503,14.088l0.097,2.495 c0.011,0.062,0.017,0.125,0.017,0.188c0,0.58-0.47,1.051-1.05,1.051c-0.004-0.001-0.008-0.001-0.012,0h-7.867 c-0.549,0-1.005-0.423-1.047-0.97l-0.202-2.623c-0.676-6.082,1.508-12.218,6.494-18.202c4.319-5.087,6.816-8.865,6.816-13.145 c0-4.829-3.036-7.536-8.548-7.624c-3.403,0-7.242,1.171-9.534,2.913c-0.264,0.201-0.607,0.264-0.925,0.173 s-0.575-0.327-0.693-0.636l-2.42-6.354c-0.169-0.442-0.02-0.943,0.364-1.224c3.538-2.573,9.441-4.235,15.041-4.235 c12.36,0,17.894,7.975,17.894,15.877C63.652,33.765,59.785,38.919,54.73,44.921z" style="fill: rgb(172, 67, 67);"/> + </g> + <rect x="-52.271" y="-25.92" width="160.339" height="87.431" style="fill: rgba(216, 216, 216, 0); stroke: rgb(172, 67, 67); stroke-width: 3px;"/> +</svg> \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-03.svg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-03.svg new file mode 100644 index 000000000..949a14f9f --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-03.svg @@ -0,0 +1,75 @@ +<?xml version="1.0" encoding="utf-8"?> +<svg viewBox="-46.534 -212.944 529.701 256.516" width="540.055px" height="256.516px" xmlns="http://www.w3.org/2000/svg" xmlns:bx="https://boxy-svg.com"> + <defs> + <style bx:fonts="Open Sans">@import url(https://fonts.googleapis.com/css2?family=Open+Sans%3Aital%2Cwght%400%2C300%3B0%2C400%3B0%2C500%3B0%2C600%3B0%2C700%3B0%2C800%3B1%2C300%3B1%2C400%3B1%2C500%3B1%2C600%3B1%2C700%3B1%2C800&display=swap);</style> + </defs> + <g id="memory-user-john-admin.svg" transform="matrix(1.7087579965591428, 0, 0, 1.7087579965591428, -443.0394287109375, -47.16891479492199)" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"> + <path id="path-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -87.403 L 333.494 -87.403 L 333.494 -57.762 L 234.068 -57.762 L 234.068 -87.403 Z" style=""/> + <path id="path-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -87.403 L 492.549 -87.403 L 492.549 -57.762 L 336.651 -57.762 L 336.651 -87.403 Z" style=""/> + <path id="path-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -53.711 L 333.494 -53.711 L 333.494 -24.07 L 234.068 -24.07 L 234.068 -53.711 Z" style=""/> + <path id="path-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -53.711 L 492.783 -53.711 L 492.783 -24.07 L 336.651 -24.07 L 336.651 -53.711 Z" style=""/> + <path id="path-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.189 -20.462 L 333.615 -20.462 L 333.615 9.179 L 234.189 9.179 L 234.189 -20.462 Z" style=""/> + <path id="path-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.791 -20.462 L 493.057 -20.462 L 493.057 9.179 L 336.791 9.179 L 336.791 -20.462 Z" style=""/> + <path id="path-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 13.088 L 333.494 13.088 L 333.494 43.59 L 234.068 43.59 L 234.068 13.088 Z" style=""/> + <path id="path-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 13.088 L 493.054 13.088 L 493.054 43.59 L 336.651 43.59 L 336.651 13.088 Z" style=""/> + <text id="<global>" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, -102.41082, 25.040792)" x="263.861" y="-70.535">key</text> + <text id="text-2" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 13.577469, 25.203112)" x="263.861" y="-70.535">value</text> + <text id="text-3" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.071654, 0, 0, 1.142641, -43.57296, 46.390514)" x="263.861" y="-70.535">image-01.jpg</text> + <text id="text-4" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.02475, 80.086914)" x="263.861" y="-70.535">image-02.jpg</text> + <text id="text-5" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.468063, 113.637062)" x="263.861" y="-70.535">image-03.jpg</text> + <g transform="matrix(0.000041, -1, 1.521843, 0.000133, 289.017426, 2.750529)" style="transform-origin: 105.999px -35.503px;"> + <path d="M 105.537 7.964 L 110.918 13.895 L 100.155 13.895 L 105.537 7.964 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 100.155 7.964 10.763 5.931 0.5 0 1@87a4267a"/> + <g> + <rect x="105.024" y="-69.436" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1.024052, -0.03572, 31.45084)" style=""> + <rect x="105.024" y="-69.051" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + </g> + <text id="text-6" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.038937, 0, 0, 1.142641, 69.710487, 42.749538)" x="263.861" y="-70.535">WeakRef object</text> + <g transform="matrix(0.441364, 0, 0, 0.412845, 420.761139, -59.372555)" style=""> + <g> + <path d=" M 58.15 23.65 L 58.15 20 54.25 20 Q 50.95 20 49.35 22.2 47.75 24.4 47.75 27.7 L 47.05 41.45 Q 46.8 43.25 46.1 44.6 45.45 46.05 44.35 47.05 43.9 47.55 43.2 47.75 42.8 47.95 41.85 48.15 L 41.85 51.6 Q 42.95 51.8 43.2 51.9 43.9 52.1 44.35 52.45 45.45 53.25 46.05 54.55 47.65 58.15 47.65 61.65 L 47.65 72.1 Q 47.65 75.7 49.45 77.85 51.25 80 54.1 80 L 58.15 80 58.15 76.25 55.25 76.25 Q 54.1 76.25 53.35 75.65 52.55 74.95 52.2 74 L 51.65 71.75 51.5 69.45 51.35 59.8 Q 51.35 57.55 50.8 55.8 50.15 53.65 49.65 52.8 48.8 51.5 47.9 50.8 47.05 50.1 46.1 49.75 47 49.3 47.8 48.7 48.95 47.85 49.6 46.75 50.1 45.8 50.75 43.7 51.35 41.85 51.35 39.45 L 51.55 28.05 52 25.9 Q 52.35 24.85 53.15 24.2 53.9 23.65 55.25 23.65 L 58.15 23.65 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <g transform="matrix(-0.441364, 0, 0, -0.412845, 414.750092, -88.816727)" style="transform-origin: 50px 50px;"> + <g> + <path d="M 14.392 23.65 L 14.392 20 L 10.492 20 C 8.292 20 6.659 20.733 5.592 22.2 C 4.525 23.667 3.992 25.5 3.992 27.7 L 3.292 41.45 C 3.125 42.65 2.809 43.7 2.342 44.6 C 1.909 45.567 1.325 46.383 0.592 47.05 C 0.292 47.383 -0.091 47.617 -0.558 47.75 C -0.825 47.883 -1.275 48.017 -1.908 48.15 L -1.908 51.6 C -1.175 51.733 -0.725 51.833 -0.558 51.9 C -0.091 52.033 0.292 52.217 0.592 52.45 C 1.325 52.983 1.892 53.683 2.292 54.55 C 3.359 56.95 3.892 59.317 3.892 61.65 L 3.892 72.1 C 3.892 74.5 4.492 76.417 5.692 77.85 C 6.892 79.283 8.442 80 10.342 80 L 14.392 80 L 14.392 76.25 L 11.492 76.25 C 10.725 76.25 10.092 76.05 9.592 75.65 C 9.059 75.183 8.675 74.633 8.442 74 L 7.892 71.75 L 7.742 69.45 L 7.592 59.8 C 7.592 58.3 7.409 56.967 7.042 55.8 C 6.609 54.367 6.225 53.367 5.892 52.8 C 5.325 51.933 4.742 51.267 4.142 50.8 C 3.575 50.333 2.975 49.983 2.342 49.75 C 2.942 49.45 3.509 49.1 4.042 48.7 C 4.809 48.133 5.409 47.483 5.842 46.75 C 6.175 46.117 6.559 45.1 6.992 43.7 C 7.392 42.467 7.592 41.05 7.592 39.45 L 7.792 28.05 L 8.242 25.9 C 8.475 25.2 8.859 24.633 9.392 24.2 C 9.892 23.833 10.592 23.65 11.492 23.65 L 14.392 23.65 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 255.632 27.46 L 268.667 47.91 L 242.597 47.91 L 255.632 27.46 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, 1, 0.000093, 86.522967, -302.258205)" bx:shape="triangle 242.597 27.46 26.07 20.45 0.5 0 1@97f9cc39"/> + <rect x="254.391" y="-239.419" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.367" y="-203.504" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.367" y="-168.53" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.391" y="-243.818" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <rect x="254.367" y="-208.398" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <rect x="254.367" y="-172.584" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <text id="text-7" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(-2.353923, 0, 0, -2.767724, 945.975708, -228.376572)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 108.749 -57.365 L 108.749 -61.015 L 104.849 -61.015 C 102.649 -61.015 101.015 -60.281 99.949 -58.815 C 98.882 -57.348 98.349 -55.515 98.349 -53.315 L 97.649 -39.565 C 97.482 -38.365 97.165 -37.315 96.699 -36.415 C 96.265 -35.448 95.682 -34.631 94.949 -33.965 C 94.649 -33.631 94.265 -33.398 93.799 -33.265 C 93.532 -33.131 93.082 -32.998 92.449 -32.865 L 92.449 -29.415 C 93.182 -29.281 93.632 -29.181 93.799 -29.115 C 94.265 -28.981 94.649 -28.798 94.949 -28.565 C 95.682 -28.031 96.249 -27.331 96.649 -26.465 C 97.715 -24.065 98.249 -21.698 98.249 -19.365 L 98.249 -8.915 C 98.249 -6.515 98.849 -4.598 100.049 -3.165 C 101.249 -1.731 102.799 -1.015 104.699 -1.015 L 108.749 -1.015 L 108.749 -4.765 L 105.849 -4.765 C 105.082 -4.765 104.449 -4.965 103.949 -5.365 C 103.415 -5.831 103.032 -6.381 102.799 -7.015 L 102.249 -9.265 L 102.099 -11.565 L 101.949 -21.215 C 101.949 -22.715 101.765 -24.048 101.399 -25.215 C 100.965 -26.648 100.582 -27.648 100.249 -28.215 C 99.682 -29.081 99.099 -29.748 98.499 -30.215 C 97.932 -30.681 97.332 -31.031 96.699 -31.265 C 97.299 -31.565 97.865 -31.915 98.399 -32.315 C 99.165 -32.881 99.765 -33.531 100.199 -34.265 C 100.532 -34.898 100.915 -35.915 101.349 -37.315 C 101.749 -38.548 101.949 -39.965 101.949 -41.565 L 102.149 -52.965 L 102.599 -55.115 C 102.832 -55.815 103.215 -56.381 103.749 -56.815 C 104.249 -57.181 104.949 -57.365 105.849 -57.365 L 108.749 -57.365 Z" style="fill: rgb(175, 110, 36); transform-box: fill-box; transform-origin: 50% 50%;" transform="matrix(-1, 0, 0, -1, -0.000004, -0.000028)"/> + <path d="M 15.325 -57.155 L 15.325 -60.805 L 11.425 -60.805 C 9.225 -60.805 7.591 -60.072 6.525 -58.605 C 5.458 -57.139 4.925 -55.305 4.925 -53.105 L 4.225 -39.355 C 4.058 -38.155 3.741 -37.105 3.275 -36.205 C 2.841 -35.239 2.258 -34.422 1.525 -33.755 C 1.225 -33.422 0.841 -33.189 0.375 -33.055 C 0.108 -32.922 -0.342 -32.789 -0.975 -32.655 L -0.975 -29.205 C -0.242 -29.072 0.208 -28.972 0.375 -28.905 C 0.841 -28.772 1.225 -28.589 1.525 -28.355 C 2.258 -27.822 2.825 -27.122 3.225 -26.255 C 4.291 -23.855 4.825 -21.489 4.825 -19.155 L 4.825 -8.705 C 4.825 -6.305 5.425 -4.389 6.625 -2.955 C 7.825 -1.522 9.375 -0.805 11.275 -0.805 L 15.325 -0.805 L 15.325 -4.555 L 12.425 -4.555 C 11.658 -4.555 11.025 -4.755 10.525 -5.155 C 9.991 -5.622 9.608 -6.172 9.375 -6.805 L 8.825 -9.055 L 8.675 -11.355 L 8.525 -21.005 C 8.525 -22.505 8.341 -23.839 7.975 -25.005 C 7.541 -26.439 7.158 -27.439 6.825 -28.005 C 6.258 -28.872 5.675 -29.539 5.075 -30.005 C 4.508 -30.472 3.908 -30.822 3.275 -31.055 C 3.875 -31.355 4.441 -31.705 4.975 -32.105 C 5.741 -32.672 6.341 -33.322 6.775 -34.055 C 7.108 -34.689 7.491 -35.705 7.925 -37.105 C 8.325 -38.339 8.525 -39.755 8.525 -41.355 L 8.725 -52.755 L 9.175 -54.905 C 9.408 -55.605 9.791 -56.172 10.325 -56.605 C 10.825 -56.972 11.525 -57.155 12.425 -57.155 L 15.325 -57.155 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + </g> + <g transform="matrix(0.3355039954185486, 0, 0, 0.3355039954185486, 67.31304168701172, -387.90008544921875)" style=""> + <g transform="matrix(0.178489, 0, 0, 0.175502, 747.077209, 776.317017)" style=""> + <path d="M 267.184 437.8 C 267.184 466.3 290.384 489.4 318.784 489.4 L 704.984 489.4 C 733.484 489.4 756.584 466.2 756.584 437.8 L 756.584 51.6 C 756.584 23.1 733.384 0 704.984 0 L 318.784 0 C 290.284 0 267.184 23.2 267.184 51.6 C 267.184 51.6 267.184 437.8 267.184 437.8 Z M 704.984 464.9 L 318.784 464.9 C 303.884 464.9 291.684 452.7 291.684 437.8 L 291.684 373.3 L 384.484 280.5 L 463.784 359.8 C 468.584 364.6 476.284 364.6 481.084 359.8 L 624.284 216.6 L 732.084 324.4 L 732.084 437.8 C 732.084 452.7 719.884 464.9 704.984 464.9 Z M 318.784 24.5 L 704.984 24.5 C 719.884 24.5 732.084 36.7 732.084 51.6 L 732.084 289.7 L 632.884 190.6 C 628.084 185.8 620.384 185.8 615.584 190.6 L 472.384 333.8 L 393.084 254.5 C 388.284 249.7 380.584 249.7 375.784 254.5 L 291.684 338.6 L 291.684 51.6 C 291.684 36.7 303.884 24.5 318.784 24.5 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 418.884 196.1 C 453.284 196.1 481.184 168.1 481.184 133.8 C 481.184 99.5 453.184 71.5 418.884 71.5 C 384.584 71.5 356.584 99.5 356.584 133.8 C 356.584 168.1 384.484 196.1 418.884 196.1 Z M 418.884 96 C 439.784 96 456.684 113 456.684 133.8 C 456.684 154.6 439.684 171.6 418.884 171.6 C 398.084 171.6 381.084 154.6 381.084 133.8 C 381.084 113 397.984 96 418.884 96 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <path d="M -161.945 -20.71 L -152.75 -5.287 L -171.141 -5.287 L -161.945 -20.71 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, -0.9999999999999999, -0.00008700000762473792, 281.6644897460937, -149.34440612792974)" bx:shape="triangle -171.141 -20.71 18.391 15.423 0.5 0 1@d86e5ee8"/> + <rect x="179.461" y="-180.566" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 323.0706176757812, 192.94052124023426)"/> + <rect x="179.444" y="-153.479" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 323.0706176757812, 192.94052124023426)"/> + <rect x="179.444" y="-127.103" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 323.0706176757812, 192.94052124023426)"/> + <rect x="179.461" y="-183.883" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 404.85729980468744, 193.00871276855457)"/> + <rect x="179.444" y="-157.171" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 404.85729980468744, 193.00871276855457)"/> + <rect x="179.444" y="-130.16" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -0.9999999999999999, 0.9999999999999999, 0.00008700000762473792, 404.85729980468744, 193.00871276855457)"/> + <text id="text-8" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.7752919197082517, 0, 0, 1.9524970054626463, -325.1152648925781, 140.82048034667957)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 318.601 -16.998 L 318.601 -19.573 L 315.66 -19.573 C 314.001 -19.573 312.769 -19.055 311.964 -18.021 C 311.16 -16.986 310.758 -15.693 310.758 -14.141 L 310.23 -4.441 C 310.104 -3.594 309.865 -2.853 309.513 -2.218 C 309.186 -1.536 308.746 -0.96 308.193 -0.49 C 307.967 -0.255 307.678 -0.09 307.326 0.004 C 307.125 0.098 306.786 0.192 306.308 0.286 L 306.308 2.72 C 306.861 2.814 307.2 2.884 307.326 2.931 C 307.678 3.025 307.967 3.155 308.193 3.319 C 308.746 3.696 309.174 4.189 309.475 4.801 C 310.28 6.494 310.682 8.163 310.682 9.81 L 310.682 17.181 C 310.682 18.875 311.135 20.227 312.04 21.238 C 312.945 22.249 314.114 22.755 315.547 22.755 L 318.601 22.755 L 318.601 20.109 L 316.414 20.109 C 315.836 20.109 315.358 19.968 314.981 19.686 C 314.579 19.357 314.29 18.969 314.114 18.522 L 313.699 16.935 L 313.586 15.312 L 313.473 8.504 C 313.473 7.446 313.334 6.506 313.058 5.683 C 312.731 4.671 312.442 3.966 312.191 3.566 C 311.763 2.955 311.323 2.485 310.871 2.155 C 310.443 1.826 309.991 1.579 309.513 1.415 C 309.966 1.203 310.393 0.956 310.795 0.674 C 311.373 0.274 311.826 -0.184 312.153 -0.702 C 312.404 -1.148 312.693 -1.866 313.02 -2.853 C 313.322 -3.723 313.473 -4.723 313.473 -5.852 L 313.623 -13.894 L 313.963 -15.41 C 314.139 -15.904 314.428 -16.304 314.83 -16.61 C 315.207 -16.868 315.735 -16.998 316.414 -16.998 L 318.601 -16.998 Z" style="fill: rgb(175, 110, 36);" transform="matrix(0.9999999999999999, 0, 0, 0.9999999999999999, 0, -1.1368683772161603e-13)"/> + <path d="M 389.058 -17.145 L 389.058 -19.72 L 386.117 -19.72 C 384.458 -19.72 383.226 -19.203 382.421 -18.168 C 381.617 -17.134 381.215 -15.84 381.215 -14.288 L 380.687 -4.588 C 380.561 -3.742 380.322 -3.001 379.97 -2.366 C 379.644 -1.684 379.204 -1.108 378.651 -0.638 C 378.424 -0.403 378.135 -0.238 377.783 -0.144 C 377.582 -0.05 377.243 0.044 376.765 0.138 L 376.765 2.572 C 377.318 2.666 377.658 2.737 377.783 2.784 C 378.135 2.878 378.424 3.007 378.651 3.172 C 379.204 3.548 379.631 4.042 379.933 4.653 C 380.737 6.346 381.139 8.016 381.139 9.662 L 381.139 17.034 C 381.139 18.727 381.592 20.079 382.497 21.09 C 383.402 22.101 384.571 22.607 386.004 22.607 L 389.058 22.607 L 389.058 19.961 L 386.871 19.961 C 386.293 19.961 385.815 19.82 385.438 19.538 C 385.036 19.209 384.747 18.821 384.571 18.374 L 384.156 16.787 L 384.043 15.164 L 383.93 8.357 C 383.93 7.299 383.792 6.358 383.515 5.535 C 383.188 4.524 382.899 3.818 382.648 3.419 C 382.22 2.807 381.78 2.337 381.328 2.008 C 380.901 1.679 380.448 1.432 379.97 1.267 C 380.423 1.055 380.85 0.808 381.253 0.526 C 381.831 0.127 382.283 -0.332 382.61 -0.849 C 382.861 -1.296 383.151 -2.013 383.477 -3.001 C 383.779 -3.871 383.93 -4.87 383.93 -5.999 L 384.081 -14.041 L 384.42 -15.558 C 384.596 -16.052 384.885 -16.452 385.287 -16.757 C 385.664 -17.016 386.192 -17.145 386.871 -17.145 L 389.058 -17.145 Z" style="fill: rgb(175, 110, 36); transform-origin: 382.912px 1.4435px;" transform="matrix(-1, 0, 0, -1, 0, -0.000015020371)"/> + <path d="M 333.236 -45.236 C 333.236 -43.558 334.625 -42.198 336.326 -42.198 L 359.453 -42.198 C 361.16 -42.198 362.543 -43.564 362.543 -45.236 L 362.543 -67.977 C 362.543 -69.655 361.154 -71.015 359.453 -71.015 L 336.326 -71.015 C 334.619 -71.015 333.236 -69.649 333.236 -67.977 C 333.236 -67.977 333.236 -45.236 333.236 -45.236 Z M 359.453 -43.641 L 336.326 -43.641 C 335.434 -43.641 334.703 -44.359 334.703 -45.236 L 334.703 -49.034 L 340.26 -54.499 L 345.009 -49.829 C 345.297 -49.547 345.758 -49.547 346.045 -49.829 L 354.62 -58.261 L 361.076 -51.914 L 361.076 -45.236 C 361.076 -44.359 360.345 -43.641 359.453 -43.641 Z M 336.326 -69.572 L 359.453 -69.572 C 360.345 -69.572 361.076 -68.854 361.076 -67.977 L 361.076 -53.957 L 355.135 -59.792 C 354.848 -60.075 354.387 -60.075 354.099 -59.792 L 345.524 -51.36 L 340.775 -56.029 C 340.488 -56.312 340.027 -56.312 339.739 -56.029 L 334.703 -51.077 L 334.703 -67.977 C 334.703 -68.854 335.434 -69.572 336.326 -69.572 Z" style="fill: rgb(175, 110, 36);" transform="matrix(0.9999999999999999, 0, 0, 0.9999999999999999, 0, -1.1368683772161603e-13)"/> + <path d="M 341.32 -59.468 C 343.38 -59.468 345.051 -61.117 345.051 -63.137 C 345.051 -65.156 343.374 -66.805 341.32 -66.805 C 339.266 -66.805 337.59 -65.156 337.59 -63.137 C 337.59 -61.117 339.26 -59.468 341.32 -59.468 Z M 341.32 -65.362 C 342.572 -65.362 343.584 -64.361 343.584 -63.137 C 343.584 -61.912 342.566 -60.911 341.32 -60.911 C 340.075 -60.911 339.057 -61.912 339.057 -63.137 C 339.057 -64.361 340.069 -65.362 341.32 -65.362 Z" style="fill: rgb(175, 110, 36);" transform="matrix(0.9999999999999999, 0, 0, 0.9999999999999999, 0, -1.1368683772161603e-13)"/> + <path d="M 333.305 13.146 C 333.305 14.824 334.694 16.184 336.395 16.184 L 359.522 16.184 C 361.229 16.184 362.612 14.818 362.612 13.146 L 362.612 -9.594 C 362.612 -11.272 361.223 -12.633 359.522 -12.633 L 336.395 -12.633 C 334.688 -12.633 333.305 -11.267 333.305 -9.594 C 333.305 -9.594 333.305 13.146 333.305 13.146 Z M 359.522 14.742 L 336.395 14.742 C 335.503 14.742 334.772 14.023 334.772 13.146 L 334.772 9.348 L 340.329 3.884 L 345.078 8.553 C 345.366 8.836 345.827 8.836 346.114 8.553 L 354.69 0.121 L 361.145 6.469 L 361.145 13.146 C 361.145 14.023 360.415 14.742 359.522 14.742 Z M 336.395 -11.19 L 359.522 -11.19 C 360.415 -11.19 361.145 -10.472 361.145 -9.594 L 361.145 4.426 L 355.205 -1.41 C 354.917 -1.692 354.456 -1.692 354.169 -1.41 L 345.593 7.022 L 340.844 2.353 C 340.557 2.07 340.096 2.07 339.809 2.353 L 334.772 7.305 L 334.772 -9.594 C 334.772 -10.472 335.503 -11.19 336.395 -11.19 Z" style="fill: rgb(175, 110, 36);" transform="matrix(0.9999999999999999, 0, 0, 0.9999999999999999, 0, -1.1368683772161603e-13)"/> + <path d="M 341.39 -1.086 C 343.45 -1.086 345.12 -2.735 345.12 -4.754 C 345.12 -6.774 343.444 -8.423 341.39 -8.423 C 339.335 -8.423 337.659 -6.774 337.659 -4.754 C 337.659 -2.735 339.329 -1.086 341.39 -1.086 Z M 341.39 -6.98 C 342.641 -6.98 343.653 -5.979 343.653 -4.754 C 343.653 -3.529 342.635 -2.528 341.39 -2.528 C 340.144 -2.528 339.126 -3.529 339.126 -4.754 C 339.126 -5.979 340.138 -6.98 341.39 -6.98 Z" style="fill: rgb(175, 110, 36);" transform="matrix(0.9999999999999999, 0, 0, 0.9999999999999999, 0, -1.1368683772161603e-13)"/> +</svg> \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-04.svg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-04.svg new file mode 100644 index 000000000..1177d6580 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-04.svg @@ -0,0 +1,77 @@ +<?xml version="1.0" encoding="utf-8"?> +<svg viewBox="-44.887 -212.944 528.054 256.516" width="540.055px" height="256.516px" xmlns="http://www.w3.org/2000/svg" xmlns:bx="https://boxy-svg.com"> + <defs> + <style bx:fonts="Open Sans">@import url(https://fonts.googleapis.com/css2?family=Open+Sans%3Aital%2Cwght%400%2C300%3B0%2C400%3B0%2C500%3B0%2C600%3B0%2C700%3B0%2C800%3B1%2C300%3B1%2C400%3B1%2C500%3B1%2C600%3B1%2C700%3B1%2C800&display=swap);</style> + </defs> + <g id="memory-user-john-admin.svg" transform="matrix(1.708757996559143, 0, 0, 1.708757996559143, -443.0394287109375, -47.168914794921875)" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"> + <text id="name:-"John"" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" style="white-space: pre;" x="19.382" y="-93.28"><tspan x="49.382" y="36.72" style="font-size: 12px; word-spacing: 0px;">name: "John"</tspan></text> + <text id="Object" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" style="white-space: pre;" x="17.598" y="-86.206"><tspan x="61.598" y="23.794" style="font-size: 14px; word-spacing: 0px;">Object</tspan></text> + <path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 40.047 -102 L 133.047 -102 L 133.047 -76 L 40.047 -76 L 40.047 -102 Z"/> + <text id="text-1" style="fill: rgb(172, 67, 67); font-family: PTMono-Regular, 'PT Mono'; font-size: 14px; white-space: pre;" transform="matrix(0.735667, 0, 0, 0.753868, 86.426659, 10.790039)" x="31.917" y="-40.191">admin</text> + <path id="path-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -87.403 L 333.494 -87.403 L 333.494 -57.762 L 234.068 -57.762 L 234.068 -87.403 Z" style=""/> + <path id="path-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -87.403 L 492.549 -87.403 L 492.549 -57.762 L 336.651 -57.762 L 336.651 -87.403 Z" style=""/> + <path id="path-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -53.711 L 333.494 -53.711 L 333.494 -24.07 L 234.068 -24.07 L 234.068 -53.711 Z" style=""/> + <path id="path-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -53.711 L 492.783 -53.711 L 492.783 -24.07 L 336.651 -24.07 L 336.651 -53.711 Z" style=""/> + <path id="path-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.189 -20.462 L 333.615 -20.462 L 333.615 9.179 L 234.189 9.179 L 234.189 -20.462 Z" style=""/> + <path id="path-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.791 -20.462 L 493.057 -20.462 L 493.057 9.179 L 336.791 9.179 L 336.791 -20.462 Z" style=""/> + <path id="path-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 13.088 L 333.494 13.088 L 333.494 43.59 L 234.068 43.59 L 234.068 13.088 Z" style=""/> + <path id="path-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 13.088 L 493.054 13.088 L 493.054 43.59 L 336.651 43.59 L 336.651 13.088 Z" style=""/> + <text id="<global>" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, -102.41082, 25.040792)" x="263.861" y="-70.535">key</text> + <text id="text-2" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 13.577469, 25.203112)" x="263.861" y="-70.535">value</text> + <text id="text-3" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.071654, 0, 0, 1.142641, -43.57296, 46.390514)" x="263.861" y="-70.535">image-01.jpg</text> + <text id="text-4" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.02475, 80.086914)" x="263.861" y="-70.535">image-02.jpg</text> + <text id="text-5" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.468063, 113.637062)" x="263.861" y="-70.535">image-03.jpg</text> + <g transform="matrix(0.000041, -1, 1.521843, 0.000133, 289.017426, 2.750529)" style="transform-origin: 105.999px -35.503px;"> + <path d="M 105.537 7.964 L 110.918 13.895 L 100.155 13.895 L 105.537 7.964 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 100.155 7.964 10.763 5.931 0.5 0 1@87a4267a"/> + <g> + <rect x="105.024" y="-69.436" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1.024052, -0.03572, 31.45084)" style=""> + <rect x="105.024" y="-69.051" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + </g> + <text id="text-6" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.038937, 0, 0, 1.142641, 69.710487, 42.749538)" x="263.861" y="-70.535">WeakRef object</text> + <g transform="matrix(0.441364, 0, 0, 0.412845, 420.761139, -59.372555)" style=""> + <g> + <path d=" M 58.15 23.65 L 58.15 20 54.25 20 Q 50.95 20 49.35 22.2 47.75 24.4 47.75 27.7 L 47.05 41.45 Q 46.8 43.25 46.1 44.6 45.45 46.05 44.35 47.05 43.9 47.55 43.2 47.75 42.8 47.95 41.85 48.15 L 41.85 51.6 Q 42.95 51.8 43.2 51.9 43.9 52.1 44.35 52.45 45.45 53.25 46.05 54.55 47.65 58.15 47.65 61.65 L 47.65 72.1 Q 47.65 75.7 49.45 77.85 51.25 80 54.1 80 L 58.15 80 58.15 76.25 55.25 76.25 Q 54.1 76.25 53.35 75.65 52.55 74.95 52.2 74 L 51.65 71.75 51.5 69.45 51.35 59.8 Q 51.35 57.55 50.8 55.8 50.15 53.65 49.65 52.8 48.8 51.5 47.9 50.8 47.05 50.1 46.1 49.75 47 49.3 47.8 48.7 48.95 47.85 49.6 46.75 50.1 45.8 50.75 43.7 51.35 41.85 51.35 39.45 L 51.55 28.05 52 25.9 Q 52.35 24.85 53.15 24.2 53.9 23.65 55.25 23.65 L 58.15 23.65 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <g transform="matrix(-0.441364, 0, 0, -0.412845, 414.750092, -88.816727)" style="transform-origin: 50px 50px;"> + <g> + <path d="M 14.392 23.65 L 14.392 20 L 10.492 20 C 8.292 20 6.659 20.733 5.592 22.2 C 4.525 23.667 3.992 25.5 3.992 27.7 L 3.292 41.45 C 3.125 42.65 2.809 43.7 2.342 44.6 C 1.909 45.567 1.325 46.383 0.592 47.05 C 0.292 47.383 -0.091 47.617 -0.558 47.75 C -0.825 47.883 -1.275 48.017 -1.908 48.15 L -1.908 51.6 C -1.175 51.733 -0.725 51.833 -0.558 51.9 C -0.091 52.033 0.292 52.217 0.592 52.45 C 1.325 52.983 1.892 53.683 2.292 54.55 C 3.359 56.95 3.892 59.317 3.892 61.65 L 3.892 72.1 C 3.892 74.5 4.492 76.417 5.692 77.85 C 6.892 79.283 8.442 80 10.342 80 L 14.392 80 L 14.392 76.25 L 11.492 76.25 C 10.725 76.25 10.092 76.05 9.592 75.65 C 9.059 75.183 8.675 74.633 8.442 74 L 7.892 71.75 L 7.742 69.45 L 7.592 59.8 C 7.592 58.3 7.409 56.967 7.042 55.8 C 6.609 54.367 6.225 53.367 5.892 52.8 C 5.325 51.933 4.742 51.267 4.142 50.8 C 3.575 50.333 2.975 49.983 2.342 49.75 C 2.942 49.45 3.509 49.1 4.042 48.7 C 4.809 48.133 5.409 47.483 5.842 46.75 C 6.175 46.117 6.559 45.1 6.992 43.7 C 7.392 42.467 7.592 41.05 7.592 39.45 L 7.792 28.05 L 8.242 25.9 C 8.475 25.2 8.859 24.633 9.392 24.2 C 9.892 23.833 10.592 23.65 11.492 23.65 L 14.392 23.65 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 255.632 27.46 L 268.667 47.91 L 242.597 47.91 L 255.632 27.46 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, 1, 0.000093, 86.522967, -302.258205)" bx:shape="triangle 242.597 27.46 26.07 20.45 0.5 0 1@97f9cc39"/> + <rect x="254.391" y="-239.419" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.367" y="-203.504" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.367" y="-168.53" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -302.258331)"/> + <rect x="254.391" y="-243.818" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <rect x="254.367" y="-208.398" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <rect x="254.367" y="-172.584" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -302.35498)"/> + <text id="text-7" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(-2.353923, 0, 0, -2.767724, 945.975708, -228.376572)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 108.749 -57.365 L 108.749 -61.015 L 104.849 -61.015 C 102.649 -61.015 101.015 -60.281 99.949 -58.815 C 98.882 -57.348 98.349 -55.515 98.349 -53.315 L 97.649 -39.565 C 97.482 -38.365 97.165 -37.315 96.699 -36.415 C 96.265 -35.448 95.682 -34.631 94.949 -33.965 C 94.649 -33.631 94.265 -33.398 93.799 -33.265 C 93.532 -33.131 93.082 -32.998 92.449 -32.865 L 92.449 -29.415 C 93.182 -29.281 93.632 -29.181 93.799 -29.115 C 94.265 -28.981 94.649 -28.798 94.949 -28.565 C 95.682 -28.031 96.249 -27.331 96.649 -26.465 C 97.715 -24.065 98.249 -21.698 98.249 -19.365 L 98.249 -8.915 C 98.249 -6.515 98.849 -4.598 100.049 -3.165 C 101.249 -1.731 102.799 -1.015 104.699 -1.015 L 108.749 -1.015 L 108.749 -4.765 L 105.849 -4.765 C 105.082 -4.765 104.449 -4.965 103.949 -5.365 C 103.415 -5.831 103.032 -6.381 102.799 -7.015 L 102.249 -9.265 L 102.099 -11.565 L 101.949 -21.215 C 101.949 -22.715 101.765 -24.048 101.399 -25.215 C 100.965 -26.648 100.582 -27.648 100.249 -28.215 C 99.682 -29.081 99.099 -29.748 98.499 -30.215 C 97.932 -30.681 97.332 -31.031 96.699 -31.265 C 97.299 -31.565 97.865 -31.915 98.399 -32.315 C 99.165 -32.881 99.765 -33.531 100.199 -34.265 C 100.532 -34.898 100.915 -35.915 101.349 -37.315 C 101.749 -38.548 101.949 -39.965 101.949 -41.565 L 102.149 -52.965 L 102.599 -55.115 C 102.832 -55.815 103.215 -56.381 103.749 -56.815 C 104.249 -57.181 104.949 -57.365 105.849 -57.365 L 108.749 -57.365 Z" style="fill: rgb(175, 110, 36); transform-box: fill-box; transform-origin: 50% 50%;" transform="matrix(-1, 0, 0, -1, -0.000004, -0.000028)"/> + <path d="M 15.325 -57.155 L 15.325 -60.805 L 11.425 -60.805 C 9.225 -60.805 7.591 -60.072 6.525 -58.605 C 5.458 -57.139 4.925 -55.305 4.925 -53.105 L 4.225 -39.355 C 4.058 -38.155 3.741 -37.105 3.275 -36.205 C 2.841 -35.239 2.258 -34.422 1.525 -33.755 C 1.225 -33.422 0.841 -33.189 0.375 -33.055 C 0.108 -32.922 -0.342 -32.789 -0.975 -32.655 L -0.975 -29.205 C -0.242 -29.072 0.208 -28.972 0.375 -28.905 C 0.841 -28.772 1.225 -28.589 1.525 -28.355 C 2.258 -27.822 2.825 -27.122 3.225 -26.255 C 4.291 -23.855 4.825 -21.489 4.825 -19.155 L 4.825 -8.705 C 4.825 -6.305 5.425 -4.389 6.625 -2.955 C 7.825 -1.522 9.375 -0.805 11.275 -0.805 L 15.325 -0.805 L 15.325 -4.555 L 12.425 -4.555 C 11.658 -4.555 11.025 -4.755 10.525 -5.155 C 9.991 -5.622 9.608 -6.172 9.375 -6.805 L 8.825 -9.055 L 8.675 -11.355 L 8.525 -21.005 C 8.525 -22.505 8.341 -23.839 7.975 -25.005 C 7.541 -26.439 7.158 -27.439 6.825 -28.005 C 6.258 -28.872 5.675 -29.539 5.075 -30.005 C 4.508 -30.472 3.908 -30.822 3.275 -31.055 C 3.875 -31.355 4.441 -31.705 4.975 -32.105 C 5.741 -32.672 6.341 -33.322 6.775 -34.055 C 7.108 -34.689 7.491 -35.705 7.925 -37.105 C 8.325 -38.339 8.525 -39.755 8.525 -41.355 L 8.725 -52.755 L 9.175 -54.905 C 9.408 -55.605 9.791 -56.172 10.325 -56.605 C 10.825 -56.972 11.525 -57.155 12.425 -57.155 L 15.325 -57.155 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <text id="text-9" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 4.89804px; font-weight: 700; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 83.283821, 89.215279)" x="263.861" y="-70.535">undefined</text> + <text id="text-10" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 4.89804px; font-weight: 700; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 82.957924, 122.90126)" x="263.861" y="-70.535">undefined</text> + </g> + <g transform="matrix(0.3355039954185486, 0, 0, 0.3355039954185486, 67.31304168701172, -387.90008544921875)" style=""> + <g transform="matrix(0.178489, 0, 0, 0.175502, 747.077209, 776.317017)" style=""> + <path d="M 267.184 437.8 C 267.184 466.3 290.384 489.4 318.784 489.4 L 704.984 489.4 C 733.484 489.4 756.584 466.2 756.584 437.8 L 756.584 51.6 C 756.584 23.1 733.384 0 704.984 0 L 318.784 0 C 290.284 0 267.184 23.2 267.184 51.6 C 267.184 51.6 267.184 437.8 267.184 437.8 Z M 704.984 464.9 L 318.784 464.9 C 303.884 464.9 291.684 452.7 291.684 437.8 L 291.684 373.3 L 384.484 280.5 L 463.784 359.8 C 468.584 364.6 476.284 364.6 481.084 359.8 L 624.284 216.6 L 732.084 324.4 L 732.084 437.8 C 732.084 452.7 719.884 464.9 704.984 464.9 Z M 318.784 24.5 L 704.984 24.5 C 719.884 24.5 732.084 36.7 732.084 51.6 L 732.084 289.7 L 632.884 190.6 C 628.084 185.8 620.384 185.8 615.584 190.6 L 472.384 333.8 L 393.084 254.5 C 388.284 249.7 380.584 249.7 375.784 254.5 L 291.684 338.6 L 291.684 51.6 C 291.684 36.7 303.884 24.5 318.784 24.5 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 418.884 196.1 C 453.284 196.1 481.184 168.1 481.184 133.8 C 481.184 99.5 453.184 71.5 418.884 71.5 C 384.584 71.5 356.584 99.5 356.584 133.8 C 356.584 168.1 384.484 196.1 418.884 196.1 Z M 418.884 96 C 439.784 96 456.684 113 456.684 133.8 C 456.684 154.6 439.684 171.6 418.884 171.6 C 398.084 171.6 381.084 154.6 381.084 133.8 C 381.084 113 397.984 96 418.884 96 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <path d="M -161.945 -20.71 L -152.75 -5.287 L -171.141 -5.287 L -161.945 -20.71 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, -1, -0.00008700000762473792, 281.66448974609375, -149.3444061279297)" bx:shape="triangle -171.141 -20.71 18.391 15.423 0.5 0 1@d86e5ee8"/> + <rect x="179.461" y="-180.566" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 323.07061767578125, 192.94052124023438)"/> + <rect x="179.444" y="-153.479" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 323.07061767578125, 192.94052124023438)"/> + <rect x="179.444" y="-127.103" width="3.4" height="13.026" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 323.07061767578125, 192.94052124023438)"/> + <rect x="179.461" y="-183.883" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 404.8572998046875, 193.0087127685547)"/> + <rect x="179.444" y="-157.171" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 404.8572998046875, 193.0087127685547)"/> + <rect x="179.444" y="-130.16" width="3.4" height="13.339" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 404.8572998046875, 193.0087127685547)"/> + <text id="text-8" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.775291919708252, 0, 0, 1.9524970054626465, -325.1152648925781, 140.8204803466797)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 318.601 -16.998 L 318.601 -19.573 L 315.66 -19.573 C 314.001 -19.573 312.769 -19.055 311.964 -18.021 C 311.16 -16.986 310.758 -15.693 310.758 -14.141 L 310.23 -4.441 C 310.104 -3.594 309.865 -2.853 309.513 -2.218 C 309.186 -1.536 308.746 -0.96 308.193 -0.49 C 307.967 -0.255 307.678 -0.09 307.326 0.004 C 307.125 0.098 306.786 0.192 306.308 0.286 L 306.308 2.72 C 306.861 2.814 307.2 2.884 307.326 2.931 C 307.678 3.025 307.967 3.155 308.193 3.319 C 308.746 3.696 309.174 4.189 309.475 4.801 C 310.28 6.494 310.682 8.163 310.682 9.81 L 310.682 17.181 C 310.682 18.875 311.135 20.227 312.04 21.238 C 312.945 22.249 314.114 22.755 315.547 22.755 L 318.601 22.755 L 318.601 20.109 L 316.414 20.109 C 315.836 20.109 315.358 19.968 314.981 19.686 C 314.579 19.357 314.29 18.969 314.114 18.522 L 313.699 16.935 L 313.586 15.312 L 313.473 8.504 C 313.473 7.446 313.334 6.506 313.058 5.683 C 312.731 4.671 312.442 3.966 312.191 3.566 C 311.763 2.955 311.323 2.485 310.871 2.155 C 310.443 1.826 309.991 1.579 309.513 1.415 C 309.966 1.203 310.393 0.956 310.795 0.674 C 311.373 0.274 311.826 -0.184 312.153 -0.702 C 312.404 -1.148 312.693 -1.866 313.02 -2.853 C 313.322 -3.723 313.473 -4.723 313.473 -5.852 L 313.623 -13.894 L 313.963 -15.41 C 314.139 -15.904 314.428 -16.304 314.83 -16.61 C 315.207 -16.868 315.735 -16.998 316.414 -16.998 L 318.601 -16.998 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 389.058 -17.145 L 389.058 -19.72 L 386.117 -19.72 C 384.458 -19.72 383.226 -19.203 382.421 -18.168 C 381.617 -17.134 381.215 -15.84 381.215 -14.288 L 380.687 -4.588 C 380.561 -3.742 380.322 -3.001 379.97 -2.366 C 379.644 -1.684 379.204 -1.108 378.651 -0.638 C 378.424 -0.403 378.135 -0.238 377.783 -0.144 C 377.582 -0.05 377.243 0.044 376.765 0.138 L 376.765 2.572 C 377.318 2.666 377.658 2.737 377.783 2.784 C 378.135 2.878 378.424 3.007 378.651 3.172 C 379.204 3.548 379.631 4.042 379.933 4.653 C 380.737 6.346 381.139 8.016 381.139 9.662 L 381.139 17.034 C 381.139 18.727 381.592 20.079 382.497 21.09 C 383.402 22.101 384.571 22.607 386.004 22.607 L 389.058 22.607 L 389.058 19.961 L 386.871 19.961 C 386.293 19.961 385.815 19.82 385.438 19.538 C 385.036 19.209 384.747 18.821 384.571 18.374 L 384.156 16.787 L 384.043 15.164 L 383.93 8.357 C 383.93 7.299 383.792 6.358 383.515 5.535 C 383.188 4.524 382.899 3.818 382.648 3.419 C 382.22 2.807 381.78 2.337 381.328 2.008 C 380.901 1.679 380.448 1.432 379.97 1.267 C 380.423 1.055 380.85 0.808 381.253 0.526 C 381.831 0.127 382.283 -0.332 382.61 -0.849 C 382.861 -1.296 383.151 -2.013 383.477 -3.001 C 383.779 -3.871 383.93 -4.87 383.93 -5.999 L 384.081 -14.041 L 384.42 -15.558 C 384.596 -16.052 384.885 -16.452 385.287 -16.757 C 385.664 -17.016 386.192 -17.145 386.871 -17.145 L 389.058 -17.145 Z" style="fill: rgb(175, 110, 36); transform-origin: 382.912px 1.4435px;" transform="matrix(-1, 0, 0, -1, 0, -0.000015020371)"/> +</svg> \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-05.svg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-05.svg new file mode 100644 index 000000000..e738f8e7e --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-05.svg @@ -0,0 +1,103 @@ +<?xml version="1.0" encoding="utf-8"?> +<svg viewBox="-56.888 -599965.495 896063.875 604140.645" width="420.466px" height="300.516px" xmlns="http://www.w3.org/2000/svg" xmlns:bx="https://boxy-svg.com"> + <defs> + <style bx:fonts="Open Sans">@import url(https://fonts.googleapis.com/css2?family=Open+Sans%3Aital%2Cwght%400%2C300%3B0%2C400%3B0%2C500%3B0%2C600%3B0%2C700%3B0%2C800%3B1%2C300%3B1%2C400%3B1%2C500%3B1%2C600%3B1%2C700%3B1%2C800&display=swap);</style> + </defs> + <g id="memory-user-john-admin.svg" transform="matrix(3405.68798828125, 0, 0, 3405.68798828125, -789862.25, -296077.03124999994)" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1" style=""> + <path id="path-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -87.403 L 333.494 -87.403 L 333.494 -57.762 L 234.068 -57.762 L 234.068 -87.403 Z" style=""/> + <path id="path-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -87.403 L 492.549 -87.403 L 492.549 -57.762 L 336.651 -57.762 L 336.651 -87.403 Z" style=""/> + <path id="path-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 -53.711 L 333.494 -53.711 L 333.494 -24.07 L 234.068 -24.07 L 234.068 -53.711 Z" style=""/> + <path id="path-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 -53.711 L 492.783 -53.711 L 492.783 -24.07 L 336.651 -24.07 L 336.651 -53.711 Z" style=""/> + <g transform="matrix(1, 0, 0, 1, 0, 42.135864)"> + <path id="path-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.189 -20.462 L 333.615 -20.462 L 333.615 9.179 L 234.189 9.179 L 234.189 -20.462 Z" style=""/> + <path id="path-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.791 -20.462 L 493.057 -20.462 L 493.057 9.179 L 336.791 9.179 L 336.791 -20.462 Z" style=""/> + <path id="path-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 234.068 13.088 L 333.494 13.088 L 333.494 43.59 L 234.068 43.59 L 234.068 13.088 Z" style=""/> + <path id="path-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M 336.651 13.088 L 493.054 13.088 L 493.054 43.59 L 336.651 43.59 L 336.651 13.088 Z" style=""/> + <text id="text-4" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.02475, 80.086914)" x="263.861" y="-70.535">image-02.jpg</text> + <text id="text-5" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.054877, 0, 0, 1.142641, -39.468063, 113.637062)" x="263.861" y="-70.535">image-03.jpg</text> + </g> + <text id="<global>" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, -102.41082, 25.040792)" x="263.861" y="-70.535">key</text> + <text id="text-2" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 11.5772px; font-weight: 600; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 13.577469, 25.203112)" x="263.861" y="-70.535">value</text> + <text id="text-3" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 14px; white-space: pre;" transform="matrix(1.071654, 0, 0, 1.142641, -43.57296, 46.390514)" x="263.861" y="-70.535">image-01.jpg</text> + <g transform="matrix(0.000041, -1, 1.521843, 0.000133, 289.017426, 2.750529)" style="transform-origin: 105.999px -35.503px;"> + <path d="M 105.537 7.964 L 110.918 13.895 L 100.155 13.895 L 105.537 7.964 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(1, 0, 0, -1, 0, 0)" bx:shape="triangle 100.155 7.964 10.763 5.931 0.5 0 1@87a4267a"/> + <g> + <rect x="105.024" y="-69.436" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <g transform="matrix(1, 0, 0, 1, 10.348016, -60.719521)"/> + <g transform="matrix(1, 0, 0, 1, 10.356147, 0.41661)"/> + <g transform="matrix(1, 0, 0, 1, 10.161898, 52.045895)"/> + <g transform="matrix(1, 0, 0, 1, -23.594774, -60.719116)"> + <rect x="79.597" y="-80.351" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-69.935" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-59.792" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.561" y="-50.176" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.551" y="-39.904" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.551" y="-29.517" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1, -23.586773, 0.417928)"> + <rect x="79.597" y="-80.351" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-69.935" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-59.792" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.561" y="-50.176" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.551" y="-39.904" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + <g transform="matrix(1, 0, 0, 1, -23.780775, 52.046886)"> + <rect x="79.798" y="-80.298" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-69.935" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.587" y="-59.792" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.561" y="-50.176" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.551" y="-39.904" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="79.551" y="-29.517" width="1.99" height="5.129" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + </g> + <g transform="matrix(1, 0, 0, 1.024052, -0.03572, 31.45084)" style=""> + <rect x="105.024" y="-69.051" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-59.02" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + <rect x="105.014" y="-48.877" width="1.99" height="5.009" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);"/> + </g> + </g> + <text id="text-6" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.038937, 0, 0, 1.142641, 69.710487, 42.749538)" x="263.861" y="-70.535">WeakRef object</text> + <g transform="matrix(0.441364, 0, 0, 0.412845, 420.761139, -59.372555)" style=""> + <g> + <path d=" M 58.15 23.65 L 58.15 20 54.25 20 Q 50.95 20 49.35 22.2 47.75 24.4 47.75 27.7 L 47.05 41.45 Q 46.8 43.25 46.1 44.6 45.45 46.05 44.35 47.05 43.9 47.55 43.2 47.75 42.8 47.95 41.85 48.15 L 41.85 51.6 Q 42.95 51.8 43.2 51.9 43.9 52.1 44.35 52.45 45.45 53.25 46.05 54.55 47.65 58.15 47.65 61.65 L 47.65 72.1 Q 47.65 75.7 49.45 77.85 51.25 80 54.1 80 L 58.15 80 58.15 76.25 55.25 76.25 Q 54.1 76.25 53.35 75.65 52.55 74.95 52.2 74 L 51.65 71.75 51.5 69.45 51.35 59.8 Q 51.35 57.55 50.8 55.8 50.15 53.65 49.65 52.8 48.8 51.5 47.9 50.8 47.05 50.1 46.1 49.75 47 49.3 47.8 48.7 48.95 47.85 49.6 46.75 50.1 45.8 50.75 43.7 51.35 41.85 51.35 39.45 L 51.55 28.05 52 25.9 Q 52.35 24.85 53.15 24.2 53.9 23.65 55.25 23.65 L 58.15 23.65 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <g transform="matrix(-0.441364, 0, 0, -0.412845, 414.750092, -88.816727)" style="transform-origin: 50px 50px;"> + <g> + <path d="M 14.392 23.65 L 14.392 20 L 10.492 20 C 8.292 20 6.659 20.733 5.592 22.2 C 4.525 23.667 3.992 25.5 3.992 27.7 L 3.292 41.45 C 3.125 42.65 2.809 43.7 2.342 44.6 C 1.909 45.567 1.325 46.383 0.592 47.05 C 0.292 47.383 -0.091 47.617 -0.558 47.75 C -0.825 47.883 -1.275 48.017 -1.908 48.15 L -1.908 51.6 C -1.175 51.733 -0.725 51.833 -0.558 51.9 C -0.091 52.033 0.292 52.217 0.592 52.45 C 1.325 52.983 1.892 53.683 2.292 54.55 C 3.359 56.95 3.892 59.317 3.892 61.65 L 3.892 72.1 C 3.892 74.5 4.492 76.417 5.692 77.85 C 6.892 79.283 8.442 80 10.342 80 L 14.392 80 L 14.392 76.25 L 11.492 76.25 C 10.725 76.25 10.092 76.05 9.592 75.65 C 9.059 75.183 8.675 74.633 8.442 74 L 7.892 71.75 L 7.742 69.45 L 7.592 59.8 C 7.592 58.3 7.409 56.967 7.042 55.8 C 6.609 54.367 6.225 53.367 5.892 52.8 C 5.325 51.933 4.742 51.267 4.142 50.8 C 3.575 50.333 2.975 49.983 2.342 49.75 C 2.942 49.45 3.509 49.1 4.042 48.7 C 4.809 48.133 5.409 47.483 5.842 46.75 C 6.175 46.117 6.559 45.1 6.992 43.7 C 7.392 42.467 7.592 41.05 7.592 39.45 L 7.792 28.05 L 8.242 25.9 C 8.475 25.2 8.859 24.633 9.392 24.2 C 9.892 23.833 10.592 23.65 11.492 23.65 L 14.392 23.65 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 255.632 27.46 L 268.667 47.91 L 242.597 47.91 L 255.632 27.46 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, 1, 0.000093, 86.522967, -404.320629)" bx:shape="triangle 242.597 27.46 26.07 20.45 0.5 0 1@97f9cc39"/> + <rect x="254.391" y="-239.419" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -404.32132)"/> + <rect x="254.367" y="-203.504" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -404.32132)"/> + <rect x="254.367" y="-168.53" width="4.82" height="17.271" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, 86.522728, -404.32132)"/> + <rect x="254.391" y="-243.818" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -404.417969)"/> + <rect x="254.367" y="-208.398" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -404.417969)"/> + <rect x="254.367" y="-172.584" width="4.82" height="17.687" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(-0.000038, 1, -1, -0.000093, -21.921179, -404.417969)"/> + <text id="text-7" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(-2.353923, 0, 0, -2.767724, 945.975708, -330.43927)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 108.749 -159.455 L 108.749 -163.105 L 104.849 -163.105 C 102.649 -163.105 101.015 -162.371 99.949 -160.905 C 98.882 -159.438 98.349 -157.605 98.349 -155.405 L 97.649 -141.655 C 97.482 -140.455 97.165 -139.405 96.699 -138.505 C 96.265 -137.538 95.682 -136.721 94.949 -136.055 C 94.649 -135.721 94.265 -135.488 93.799 -135.355 C 93.532 -135.221 93.082 -135.088 92.449 -134.955 L 92.449 -131.505 C 93.182 -131.371 93.632 -131.271 93.799 -131.205 C 94.265 -131.071 94.649 -130.888 94.949 -130.655 C 95.682 -130.121 96.249 -129.421 96.649 -128.555 C 97.715 -126.155 98.249 -123.788 98.249 -121.455 L 98.249 -111.005 C 98.249 -108.605 98.849 -106.688 100.049 -105.255 C 101.249 -103.821 102.799 -103.105 104.699 -103.105 L 108.749 -103.105 L 108.749 -106.855 L 105.849 -106.855 C 105.082 -106.855 104.449 -107.055 103.949 -107.455 C 103.415 -107.921 103.032 -108.471 102.799 -109.105 L 102.249 -111.355 L 102.099 -113.655 L 101.949 -123.305 C 101.949 -124.805 101.765 -126.138 101.399 -127.305 C 100.965 -128.738 100.582 -129.738 100.249 -130.305 C 99.682 -131.171 99.099 -131.838 98.499 -132.305 C 97.932 -132.771 97.332 -133.121 96.699 -133.355 C 97.299 -133.655 97.865 -134.005 98.399 -134.405 C 99.165 -134.971 99.765 -135.621 100.199 -136.355 C 100.532 -136.988 100.915 -138.005 101.349 -139.405 C 101.749 -140.638 101.949 -142.055 101.949 -143.655 L 102.149 -155.055 L 102.599 -157.205 C 102.832 -157.905 103.215 -158.471 103.749 -158.905 C 104.249 -159.271 104.949 -159.455 105.849 -159.455 L 108.749 -159.455 Z" style="fill: rgb(175, 110, 36); transform-box: fill-box; transform-origin: 50% 50%;" transform="matrix(-1, 0, 0, -1, -0.000041, -0.000044)"/> + <path d="M 15.325 -159.251 L 15.325 -162.901 L 11.425 -162.901 C 9.225 -162.901 7.591 -162.168 6.525 -160.701 C 5.458 -159.235 4.925 -157.401 4.925 -155.201 L 4.225 -141.451 C 4.058 -140.251 3.741 -139.201 3.275 -138.301 C 2.841 -137.335 2.258 -136.518 1.525 -135.851 C 1.225 -135.518 0.841 -135.285 0.375 -135.151 C 0.108 -135.018 -0.342 -134.885 -0.975 -134.751 L -0.975 -131.301 C -0.242 -131.168 0.208 -131.068 0.375 -131.001 C 0.841 -130.868 1.225 -130.685 1.525 -130.451 C 2.258 -129.918 2.825 -129.218 3.225 -128.351 C 4.291 -125.951 4.825 -123.585 4.825 -121.251 L 4.825 -110.801 C 4.825 -108.401 5.425 -106.485 6.625 -105.051 C 7.825 -103.618 9.375 -102.901 11.275 -102.901 L 15.325 -102.901 L 15.325 -106.651 L 12.425 -106.651 C 11.658 -106.651 11.025 -106.851 10.525 -107.251 C 9.991 -107.718 9.608 -108.268 9.375 -108.901 L 8.825 -111.151 L 8.675 -113.451 L 8.525 -123.101 C 8.525 -124.601 8.341 -125.935 7.975 -127.101 C 7.541 -128.535 7.158 -129.535 6.825 -130.101 C 6.258 -130.968 5.675 -131.635 5.075 -132.101 C 4.508 -132.568 3.908 -132.918 3.275 -133.151 C 3.875 -133.451 4.441 -133.801 4.975 -134.201 C 5.741 -134.768 6.341 -135.418 6.775 -136.151 C 7.108 -136.785 7.491 -137.801 7.925 -139.201 C 8.325 -140.435 8.525 -141.851 8.525 -143.451 L 8.725 -154.851 L 9.175 -157.001 C 9.408 -157.701 9.791 -158.268 10.325 -158.701 C 10.825 -159.068 11.525 -159.251 12.425 -159.251 L 15.325 -159.251 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <text id="text-9" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 4.89804px; font-weight: 700; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 83.283821, 131.351151)" x="263.861" y="-70.535">undefined</text> + <text id="text-10" style="fill: rgb(175, 110, 36); font-family: 'Open Sans'; font-size: 4.89804px; font-weight: 700; line-height: 19.9484px; white-space: pre;" transform="matrix(1.374145, 0, 0, 1.314286, 82.957924, 165.037125)" x="263.861" y="-70.535">undefined</text> + <text id="text-11" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(1.038937, 0, 0, 1.142641, -18.813889, 90.723404)" x="263.861" y="-70.535">Deleted by FinalizationRegistry cleanup callback</text> + </g> + <g transform="matrix(668.68603515625, 0, 0, 668.68603515625, 227309.90625, -975180.6875)" style=""> + <g transform="matrix(0.178489, 0, 0, 0.175502, 747.077209, 776.317017)" style=""> + <path d="M 267.184 437.8 C 267.184 466.3 290.384 489.4 318.784 489.4 L 704.984 489.4 C 733.484 489.4 756.584 466.2 756.584 437.8 L 756.584 51.6 C 756.584 23.1 733.384 0 704.984 0 L 318.784 0 C 290.284 0 267.184 23.2 267.184 51.6 C 267.184 51.6 267.184 437.8 267.184 437.8 Z M 704.984 464.9 L 318.784 464.9 C 303.884 464.9 291.684 452.7 291.684 437.8 L 291.684 373.3 L 384.484 280.5 L 463.784 359.8 C 468.584 364.6 476.284 364.6 481.084 359.8 L 624.284 216.6 L 732.084 324.4 L 732.084 437.8 C 732.084 452.7 719.884 464.9 704.984 464.9 Z M 318.784 24.5 L 704.984 24.5 C 719.884 24.5 732.084 36.7 732.084 51.6 L 732.084 289.7 L 632.884 190.6 C 628.084 185.8 620.384 185.8 615.584 190.6 L 472.384 333.8 L 393.084 254.5 C 388.284 249.7 380.584 249.7 375.784 254.5 L 291.684 338.6 L 291.684 51.6 C 291.684 36.7 303.884 24.5 318.784 24.5 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 418.884 196.1 C 453.284 196.1 481.184 168.1 481.184 133.8 C 481.184 99.5 453.184 71.5 418.884 71.5 C 384.584 71.5 356.584 99.5 356.584 133.8 C 356.584 168.1 384.484 196.1 418.884 196.1 Z M 418.884 96 C 439.784 96 456.684 113 456.684 133.8 C 456.684 154.6 439.684 171.6 418.884 171.6 C 398.084 171.6 381.084 154.6 381.084 133.8 C 381.084 113 397.984 96 418.884 96 Z" style="fill: rgb(175, 110, 36);"/> + </g> + </g> + <path d="M 359424.367 41276.336 L 377751.896 72015.889 L 341096.837 72015.889 L 359424.367 41276.336 Z" style="fill-rule: nonzero; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, -1, -0.00008700000762473792, 737054.1250000001, 325983.1250000001)" bx:shape="triangle 341096.837 41276.336 36655.059 30739.553 0.5 0 1@d50babe8"/> + <rect x="357681.08" y="-359881.579" width="6776.286" height="25963.08" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 737054.6250000001, 325981.3125000001)"/> + <rect x="357644.501" y="-305895.331" width="6776.286" height="25963.08" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 737054.6250000001, 325981.3125000001)"/> + <rect x="357644.501" y="-253326.04" width="6776.286" height="25963.08" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 737054.6250000001, 325981.3125000001)"/> + <rect x="357681.08" y="-366492.978" width="6776.286" height="26586.395" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 900061.8125000001, 326117.2187500001)"/> + <rect x="357644.501" y="-313254.567" width="6776.286" height="26586.395" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 900061.8125000001, 326117.2187500001)"/> + <rect x="357644.501" y="-259418.129" width="6776.286" height="26586.395" style="paint-order: stroke; stroke: rgb(53, 81, 138); stroke-width: 0px; fill: rgb(172, 67, 67);" transform="matrix(0.00004100000296602957, -1, 1, 0.00008700000762473792, 900061.8125000001, 326117.2187500001)"/> + <text id="text-8" style="fill: rgb(172, 67, 67); font-family: 'Open Sans'; font-size: 9.73113px; font-style: italic; white-space: pre;" transform="matrix(3538.295166015625, 0, 0, 3891.478759765625, -554830.25, 222101.96875)" x="263.861" y="-70.535">WeakRef object</text> + <path d="M 728146.338 -92443.632 L 728146.338 -97574.712 L 722284.793 -97574.712 C 718978.199 -97574.712 716523.314 -96542.53 714919.193 -94480.927 C 713316.016 -92418.404 712514.341 -89841.53 712514.341 -86748.938 L 711462.052 -67415.4 C 711210.096 -65727.244 710734.892 -64250.108 710033.283 -62985.482 C 709382.304 -61626.233 708504.597 -60478.003 707402.151 -59541.042 C 706951.752 -59072.499 706375.587 -58743.915 705673.655 -58556.036 C 705273.289 -58368.007 704596.909 -58181.644 703645.158 -57993.168 L 703645.158 -53141.856 C 704747.605 -52953.853 705423.812 -52816.007 705673.655 -52721.682 C 706375.587 -52533.653 706951.752 -52275.956 707402.151 -51947.819 C 708504.597 -51196.774 709359.189 -50215.843 709957.252 -48995.062 C 711561.373 -45621.509 712363.794 -42293.888 712363.794 -39012.547 L 712363.794 -24320.941 C 712363.794 -20946.169 713265.063 -18250.364 715069.292 -16236.209 C 716874.118 -14219.916 719201.895 -13211.347 722059.73 -13211.347 L 728146.338 -13211.347 L 728146.338 -18485.095 L 723786.735 -18485.095 C 722635.597 -18485.095 721683.847 -18767.275 720932.205 -19327.881 C 720129.784 -19984.452 719554.812 -20757.843 719201.895 -21648.25 L 718377.253 -24810.511 L 718150.525 -28045.597 L 717925.463 -41615.096 C 717925.463 -43722.656 717648.726 -45597.921 717099.007 -47236.193 C 716447.133 -49254.872 715870.968 -50660.078 715370.511 -51456.161 C 714517.162 -52674.979 713641.12 -53612.089 712740.149 -54268.661 C 711885.582 -54925.978 710986.4 -55416.741 710033.283 -55743.536 C 710935.472 -56166.271 711786.261 -56657.059 712587.937 -57219.753 C 713740.267 -58016.755 714644.122 -58930.278 715294.33 -59961.987 C 715794.787 -60852.37 716371.101 -62282.655 717022.975 -64250.108 C 717625.611 -65984.196 717925.463 -67976.901 717925.463 -70227.453 L 718225.811 -86255.615 L 718902.938 -89277.444 C 719252.823 -90261.705 719829.908 -91059.9 720629.445 -91669.769 C 721380.938 -92183.673 722434.122 -92443.632 723786.735 -92443.632 L 728146.338 -92443.632 Z" style="fill: rgb(175, 110, 36);"/> + <path d="M 868579.948 -92734.348 L 868579.948 -97866.322 L 862717.955 -97866.322 C 859411.336 -97866.322 856954.786 -96837.173 855350.69 -94775.421 C 853747.488 -92713.793 852947.926 -90134.359 852947.926 -87041.916 L 851895.661 -67707.906 C 851643.706 -66022.185 851168.054 -64545.67 850465.228 -63278.459 C 849816.809 -61920.428 848939.872 -60773.094 847838.32 -59836.43 C 847385.039 -59366.993 846807.506 -59039.453 846107.264 -58850.977 C 845705.979 -58662.948 845032.333 -58476.734 844077.55 -58288.705 L 844077.55 -53438.165 C 845180.295 -53250.136 845856.974 -53108.363 846107.264 -53014.659 C 846807.506 -52828.296 847385.039 -52570.897 847838.32 -52242.462 C 848939.872 -51491.715 849788.996 -50508.374 850393.421 -49291.196 C 851994.957 -45916.599 852795.738 -42588.531 852795.738 -39308.084 L 852795.738 -24614.217 C 852795.738 -21238.7 853699.12 -18545.753 855503.051 -16529.485 C 857306.038 -14515.33 859635.504 -13506.885 862491.674 -13506.885 L 868579.948 -13506.885 L 868579.948 -18781.23 L 864220.319 -18781.23 C 863067.542 -18781.23 862114.424 -19061.47 861364.149 -19624.164 C 860562.474 -20278.623 859988.421 -21052.486 859635.504 -21943.639 L 858809.496 -25105.899 L 858584.135 -28341.01 L 858357.407 -41908.844 C 858357.407 -44017.15 858082.336 -45890.452 857532.617 -47532.477 C 856880.42 -49546.483 856302.887 -50953.801 855803.201 -51749.114 C 854950.299 -52969.622 854074.73 -53906.708 853171.77 -54561.937 C 852320.832 -55217.589 851420.01 -55710.191 850465.228 -56038.154 C 851368.61 -56462.579 852219.573 -56952.447 853024.106 -57514.843 C 854174.772 -58310.926 855075.594 -59225.667 855727.194 -60256.63 C 856228.546 -61147.784 856806.351 -62576.08 857456.585 -64545.67 C 858057.531 -66279.609 858357.407 -68270.177 858357.407 -70520.281 L 858660.316 -86548.145 L 859335.33 -89573.007 C 859685.662 -90557.094 860263.045 -91354.096 861063.055 -91962.3 C 861814.548 -92479.061 862866.066 -92734.348 864220.319 -92734.348 L 868579.948 -92734.348 Z" style="fill: rgb(175, 110, 36); transform-origin: 856323px -55686.3px;" transform="matrix(-1, 0, 0, -1, 0.87500000055, -0.046875000444)"/> + <rect x="3810.427" y="-226324.344" width="889802.927" height="226474.243" style="stroke: rgb(0, 0, 0); stroke-opacity: 0; fill: rgb(172, 67, 67); opacity: 0.76;"/> +</svg> \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-01.png b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-01.png new file mode 100644 index 000000000..fc33a023a Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-01.png differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-02.png b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-02.png new file mode 100644 index 000000000..7d8bb01e8 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-02.png differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-03.gif b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-03.gif new file mode 100644 index 000000000..b81966dda Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-03.gif differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-04.jpg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-04.jpg new file mode 100644 index 000000000..ba60f1e86 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-04.jpg differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-05.gif b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-05.gif new file mode 100644 index 000000000..d34bda4d7 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-05.gif differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-06.jpg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-06.jpg new file mode 100644 index 000000000..b2655540f Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-06.jpg differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-07.gif b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-07.gif new file mode 100644 index 000000000..51f874518 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-07.gif differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-08.jpg b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-08.jpg new file mode 100644 index 000000000..5f98aec14 Binary files /dev/null and b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry-demo-08.jpg differ diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.css b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.css new file mode 100644 index 000000000..e6c9e3960 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.css @@ -0,0 +1,285 @@ +:root { + --mineralGreen: 60, 98, 85; + --viridianGreen: 97, 135, 110; + --swampGreen: 166, 187, 141; + --fallGreen: 234, 231, 177; + --brinkPink: #FA7070; + --silverChalice: 178, 178, 178; + --white: 255, 255, 255; + --black: 0, 0, 0; + + --topBarHeight: 64px; + --itemPadding: 32px; + --containerGap: 8px; +} + +@keyframes zoom-in { + 0% { + transform: scale(1, 1); + } + + 100% { + transform: scale(1.30, 1.30); + } +} + +body, html { + margin: 0; + padding: 0; +} + +.app { + min-height: 100vh; + background-color: rgba(var(--viridianGreen), 0.5); +} + +.header { + height: var(--topBarHeight); + padding: 0 24px; + display: flex; + justify-content: space-between; + align-items: center; + background-color: rgba(var(--mineralGreen), 1); +} + +.header-text { + color: white; +} + +.container { + display: flex; + gap: 24px; + padding: var(--itemPadding); +} + +.item { + width: 50%; +} + +.item--scrollable { + overflow-y: scroll; + height: calc(100vh - var(--topBarHeight) - (var(--itemPadding) * 2)); +} + +.thumbnails-container { + display: flex; + flex-wrap: wrap; + gap: 8px; + justify-content: center; + align-items: center; +} + +.thumbnail-item { + width: calc(25% - var(--containerGap)); + cursor: pointer; + position: relative; +} + +.thumbnail-item:hover { + z-index: 1; + animation: zoom-in 0.1s forwards; +} + +.thumbnail-item--selected { + outline: 3px solid rgba(var(--fallGreen), 1); + outline-offset: -3px; +} + +.badge { + width: 16px; + height: 16px; + display: flex; + justify-content: center; + align-items: center; + padding: 4px; + position: absolute; + right: 8px; + bottom: 8px; + border-radius: 50%; + border: 2px solid rgba(var(--fallGreen), 1); + background-color: rgba(var(--swampGreen), 1); +} + +.check { + display: inline-block; + transform: rotate(45deg); + border-bottom: 2px solid white; + border-right: 2px solid white; + width: 6px; + height: 12px; +} + +.img { + width: 100%; + height: 100%; + object-fit: cover; +} + +.actions { + display: flex; + flex-wrap: wrap; + justify-content: center; + align-content: center; + padding: 0 0 16px 0; + gap: 8px; +} + +.select { + padding: 16px; + cursor: pointer; + font-weight: 700; + color: rgba(var(--black), 1); + border: 2px solid rgba(var(--swampGreen), 0.5); + background-color: rgba(var(--swampGreen), 1); +} + +.select:disabled { + cursor: not-allowed; + background-color: rgba(var(--silverChalice), 1); + color: rgba(var(--black), 0.5); + border: 2px solid rgba(var(--black), 0.25); +} + +.btn { + outline: none; + padding: 16px; + cursor: pointer; + font-weight: 700; + color: rgba(var(--black), 1); + border: 2px solid rgba(var(--black), 0.5); +} + +.btn--primary { + background-color: rgba(var(--mineralGreen), 1); +} + +.btn--primary:hover:not([disabled]) { + background-color: rgba(var(--mineralGreen), 0.85); +} + +.btn--secondary { + background-color: rgba(var(--viridianGreen), 0.5); +} + +.btn--secondary:hover:not([disabled]) { + background-color: rgba(var(--swampGreen), 0.25); +} + +.btn--success { + background-color: rgba(var(--fallGreen), 1); +} + +.btn--success:hover:not([disabled]) { + background-color: rgba(var(--fallGreen), 0.85); +} + +.btn:disabled { + cursor: not-allowed; + background-color: rgba(var(--silverChalice), 1); + color: rgba(var(--black), 0.5); + border: 2px solid rgba(var(--black), 0.25); +} + +.previewContainer { + margin-bottom: 16px; + display: flex; + width: 100%; + height: 40vh; + overflow: scroll; + border: 3px solid rgba(var(--black), 1); +} + +.previewContainer--disabled { + background-color: rgba(var(--black), 0.1); + cursor: not-allowed; +} + +.canvas { + margin: auto; + display: none; +} + +.canvas--ready { + display: block; +} + +.spinnerContainer { + display: flex; + gap: 8px; + flex-direction: column; + align-content: center; + align-items: center; + margin: auto; +} + +.spinnerContainer--hidden { + display: none; +} + +.spinnerText { + margin: 0; + color: rgba(var(--mineralGreen), 1); +} + +.spinner { + display: inline-block; + width: 50px; + height: 50px; + margin: auto; + border: 3px solid rgba(var(--mineralGreen), 0.3); + border-radius: 50%; + border-top-color: rgba(var(--mineralGreen), 0.9); + animation: spin 1s ease-in-out infinite; +} + +@keyframes spin { + to { + transform: rotate(360deg); + } +} + +.loggerContainer { + display: flex; + flex-direction: column; + gap: 8px; + padding: 0 8px 8px 8px; + width: 100%; + min-height: 30vh; + max-height: 30vh; + overflow: scroll; + border-left: 3px solid rgba(var(--black), 0.25); +} + +.logger-title { + display: flex; + align-items: center; + padding: 8px; + position: sticky; + height: 40px; + min-height: 40px; + top: 0; + left: 0; + background-color: rgba(var(--viridianGreen), 1); + font-size: 24px; + font-weight: 700; + margin: 0; +} + +.logger-item { + font-size: 14px; + padding: 8px; + border: 2px solid #5a5a5a; + color: white; +} + +.logger--primary { + background-color: #13315a; +} + +.logger--success { + background-color: #385a4e; +} + +.logger--error { + background-color: #5a1a24; +} \ No newline at end of file diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.html b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.html new file mode 100644 index 000000000..7ce52f927 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.html @@ -0,0 +1,49 @@ +<!DOCTYPE HTML> +<html lang="en"> + +<head> + <meta charset="utf-8"> + <link rel="stylesheet" href="index.css"> + <title>Photo Library Collage</title> +</head> + +<body> + +<div class="app"> + <header class="header"> + <h1 class="header-text"> + Photo Library Collage + </h1> + </header> + <div class="container"> + <div class="item item--scrollable"> + <!--Thumbnails--> + <div class="thumbnails-container"></div> + </div> + <div class="item"> + <div> + <div class=actions> + <select class="select"></select> + <button class="btn btn--primary btn-create-collage"> Create collage </button> + <button class="btn btn--secondary btn-start-over"> Start over </button> + <button class="btn btn--success btn-download" onClick={downloadCollage}> Download </button> + </div> + <div class="previewContainer"> + <div class="spinnerContainer spinnerContainer--hidden"> + <div class="spinner"></div> + <p class="spinnerText"></p> + </div> + <canvas class="canvas"></canvas> + </div> + <div class="loggerContainer"> + <p class="logger-title">Logger:</p> + </div> + </div> + </div> + </div> +</div> + + +<script type="module" src="index.js"></script> +</body> +</html> diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.js b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.js new file mode 100644 index 000000000..983b34d9a --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/index.js @@ -0,0 +1,228 @@ +import { + createImageFile, + loadImage, + weakRefCache, + LAYOUTS, + images, + THUMBNAIL_PARAMS, + stateObj, +} from "./utils.js"; + +export const state = new Proxy(stateObj, { + set(target, property, value) { + const previousValue = target[property]; + + target[property] = value; + + if (previousValue !== value) { + handleStateChange(target); + } + + return true; + }, +}); + +// Elements. +const thumbnailsContainerEl = document.querySelector(".thumbnails-container"); +const selectEl = document.querySelector(".select"); +const previewContainerEl = document.querySelector(".previewContainer"); +const canvasEl = document.querySelector(".canvas"); +const createCollageBtn = document.querySelector(".btn-create-collage"); +const startOverBtn = document.querySelector(".btn-start-over"); +const downloadBtn = document.querySelector(".btn-download"); +const spinnerContainerEl = document.querySelector(".spinnerContainer"); +const spinnerTextEl = document.querySelector(".spinnerText"); +const loggerContainerEl = document.querySelector(".loggerContainer"); + +// Renders. +// Render thumbnails previews. +images.forEach((img) => { + const thumbnail = document.createElement("div"); + thumbnail.classList.add("thumbnail-item"); + + thumbnail.innerHTML = ` + <img src='${img.img}?${THUMBNAIL_PARAMS}' class="img"> + `; + + thumbnail.addEventListener("click", (e) => handleSelection(e, img)); + + thumbnailsContainerEl.appendChild(thumbnail); +}); +// Render layouts select. +LAYOUTS.forEach((layout) => { + const option = document.createElement("option"); + option.value = JSON.stringify(layout); + option.innerHTML = layout.name; + selectEl.appendChild(option); +}); + +const handleStateChange = (state) => { + if (state.loading) { + selectEl.disabled = true; + createCollageBtn.disabled = true; + startOverBtn.disabled = true; + downloadBtn.disabled = true; + previewContainerEl.classList.add("previewContainer--disabled"); + spinnerContainerEl.classList.remove("spinnerContainer--hidden"); + spinnerTextEl.innerText = "Loading..."; + canvasEl.classList.remove("canvas--ready"); + } else if (!state.loading) { + selectEl.disabled = false; + createCollageBtn.disabled = false; + startOverBtn.disabled = false; + downloadBtn.disabled = false; + previewContainerEl.classList.remove("previewContainer--disabled"); + spinnerContainerEl.classList.add("spinnerContainer--hidden"); + canvasEl.classList.add("canvas--ready"); + } + + if (!state.selectedImages.size) { + createCollageBtn.disabled = true; + document.querySelectorAll(".badge").forEach((item) => item.remove()); + } else if (state.selectedImages.size && !state.loading) { + createCollageBtn.disabled = false; + } + + if (!state.collageRendered) { + downloadBtn.disabled = true; + } else if (state.collageRendered) { + downloadBtn.disabled = false; + } +}; +handleStateChange(state); + +const handleSelection = (e, imgName) => { + const imgEl = e.currentTarget; + + imgEl.classList.toggle("thumbnail-item--selected"); + + if (state.selectedImages.has(imgName)) { + state.selectedImages.delete(imgName); + state.selectedImages = new Set(state.selectedImages); + imgEl.querySelector(".badge")?.remove(); + } else { + state.selectedImages = new Set(state.selectedImages.add(imgName)); + + const badge = document.createElement("div"); + badge.classList.add("badge"); + badge.innerHTML = ` + <div class="check" /> + `; + imgEl.prepend(badge); + } +}; + +// Make a wrapper function. +let getCachedImage; +(async () => { + getCachedImage = await weakRefCache(loadImage); +})(); + +const calculateGridRows = (blobsLength) => + Math.ceil(blobsLength / state.currentLayout.columns); + +const drawCollage = (images) => { + state.drawing = true; + + let context = canvasEl.getContext("2d"); + + /** + * Calculate canvas dimensions based on the current layout. + * */ + context.canvas.width = + state.currentLayout.itemWidth * state.currentLayout.columns; + context.canvas.height = + calculateGridRows(images.length) * state.currentLayout.itemHeight; + + let currentRow = 0; + let currentCanvasDx = 0; + let currentCanvasDy = 0; + + for (let i = 0; i < images.length; i++) { + /** + * Get current row of the collage. + * */ + if (i % state.currentLayout.columns === 0) { + currentRow += 1; + currentCanvasDx = 0; + + if (currentRow > 1) { + currentCanvasDy += state.currentLayout.itemHeight; + } + } + + context.drawImage( + images[i], + 0, + 0, + images[i].width, + images[i].height, + currentCanvasDx, + currentCanvasDy, + state.currentLayout.itemWidth, + state.currentLayout.itemHeight, + ); + + currentCanvasDx += state.currentLayout.itemWidth; + } + + state.drawing = false; + state.collageRendered = true; +}; + +const createCollage = async () => { + state.loading = true; + + const images = []; + + for (const image of state.selectedImages.values()) { + const blobImage = await getCachedImage(image.img); + + const url = URL.createObjectURL(blobImage); + const img = await createImageFile(url); + + images.push(img); + URL.revokeObjectURL(url); + } + + state.loading = false; + + drawCollage(images); +}; + +/** + * Clear all settled data to start over. + * */ +const startOver = () => { + state.selectedImages = new Set(); + state.collageRendered = false; + const context = canvasEl.getContext("2d"); + context.clearRect(0, 0, canvasEl.width, canvasEl.height); + + document + .querySelectorAll(".thumbnail-item--selected") + .forEach((item) => item.classList.remove("thumbnail-item--selected")); + + loggerContainerEl.innerHTML = '<p class="logger-title">Logger:</p>'; +}; + +const downloadCollage = () => { + const date = new Date(); + const fileName = `Collage-${date.getDay()}-${date.getMonth()}-${date.getFullYear()}.png`; + const img = canvasEl.toDataURL("image/png"); + const link = document.createElement("a"); + link.download = fileName; + link.href = img; + link.click(); + link.remove(); +}; + +const changeLayout = ({ target }) => { + state.currentLayout = JSON.parse(target.value); +}; + +// Listeners. +selectEl.addEventListener("change", changeLayout); +createCollageBtn.addEventListener("click", createCollage); +startOverBtn.addEventListener("click", startOver); +downloadBtn.addEventListener("click", downloadCollage); diff --git a/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/utils.js b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/utils.js new file mode 100644 index 000000000..f0140c116 --- /dev/null +++ b/1-js/99-js-misc/07-weakref-finalizationregistry/weakref-finalizationregistry.view/utils.js @@ -0,0 +1,321 @@ +const loggerContainerEl = document.querySelector(".loggerContainer"); + +export const images = [ + { + img: "https://images.unsplash.com/photo-1471357674240-e1a485acb3e1", + }, + { + img: "https://images.unsplash.com/photo-1589118949245-7d38baf380d6", + }, + { + img: "https://images.unsplash.com/photo-1527631746610-bca00a040d60", + }, + { + img: "https://images.unsplash.com/photo-1500835556837-99ac94a94552", + }, + { + img: "https://images.unsplash.com/photo-1503220317375-aaad61436b1b", + }, + { + img: "https://images.unsplash.com/photo-1501785888041-af3ef285b470", + }, + { + img: "https://images.unsplash.com/photo-1528543606781-2f6e6857f318", + }, + { + img: "https://images.unsplash.com/photo-1523906834658-6e24ef2386f9", + }, + { + img: "https://images.unsplash.com/photo-1539635278303-d4002c07eae3", + }, + { + img: "https://images.unsplash.com/photo-1533105079780-92b9be482077", + }, + { + img: "https://images.unsplash.com/photo-1516483638261-f4dbaf036963", + }, + { + img: "https://images.unsplash.com/photo-1502791451862-7bd8c1df43a7", + }, + { + img: "https://plus.unsplash.com/premium_photo-1663047367140-91adf819d007", + }, + { + img: "https://images.unsplash.com/photo-1506197603052-3cc9c3a201bd", + }, + { + img: "https://images.unsplash.com/photo-1517760444937-f6397edcbbcd", + }, + { + img: "https://images.unsplash.com/photo-1518684079-3c830dcef090", + }, + { + img: "https://images.unsplash.com/photo-1505832018823-50331d70d237", + }, + { + img: "https://images.unsplash.com/photo-1524850011238-e3d235c7d4c9", + }, + { + img: "https://plus.unsplash.com/premium_photo-1661277758451-b5053309eea1", + }, + { + img: "https://images.unsplash.com/photo-1541410965313-d53b3c16ef17", + }, + { + img: "https://images.unsplash.com/photo-1528702748617-c64d49f918af", + }, + { + img: "https://images.unsplash.com/photo-1502003148287-a82ef80a6abc", + }, + { + img: "https://plus.unsplash.com/premium_photo-1661281272544-5204ea3a481a", + }, + { + img: "https://images.unsplash.com/photo-1503457574462-bd27054394c1", + }, + { + img: "https://images.unsplash.com/photo-1499363536502-87642509e31b", + }, + { + img: "https://images.unsplash.com/photo-1551918120-9739cb430c6d", + }, + { + img: "https://plus.unsplash.com/premium_photo-1661382219642-43e54f7e81d7", + }, + { + img: "https://images.unsplash.com/photo-1497262693247-aa258f96c4f5", + }, + { + img: "https://images.unsplash.com/photo-1525254134158-4fd5fdd45793", + }, + { + img: "https://plus.unsplash.com/premium_photo-1661274025419-4c54107d5c48", + }, + { + img: "https://images.unsplash.com/photo-1553697388-94e804e2f0f6", + }, + { + img: "https://images.unsplash.com/photo-1574260031597-bcd9eb192b4f", + }, + { + img: "https://images.unsplash.com/photo-1536323760109-ca8c07450053", + }, + { + img: "https://images.unsplash.com/photo-1527824404775-dce343118ebc", + }, + { + img: "https://images.unsplash.com/photo-1612278675615-7b093b07772d", + }, + { + img: "https://images.unsplash.com/photo-1522010675502-c7b3888985f6", + }, + { + img: "https://images.unsplash.com/photo-1501555088652-021faa106b9b", + }, + { + img: "https://plus.unsplash.com/premium_photo-1669223469435-27e091439169", + }, + { + img: "https://images.unsplash.com/photo-1506012787146-f92b2d7d6d96", + }, + { + img: "https://images.unsplash.com/photo-1511739001486-6bfe10ce785f", + }, + { + img: "https://images.unsplash.com/photo-1553342385-111fd6bc6ab3", + }, + { + img: "https://images.unsplash.com/photo-1516546453174-5e1098a4b4af", + }, + { + img: "https://images.unsplash.com/photo-1527142879-95b61a0b8226", + }, + { + img: "https://images.unsplash.com/photo-1520466809213-7b9a56adcd45", + }, + { + img: "https://images.unsplash.com/photo-1516939884455-1445c8652f83", + }, + { + img: "https://images.unsplash.com/photo-1545389336-cf090694435e", + }, + { + img: "https://plus.unsplash.com/premium_photo-1669223469455-b7b734c838f4", + }, + { + img: "https://images.unsplash.com/photo-1454391304352-2bf4678b1a7a", + }, + { + img: "https://images.unsplash.com/photo-1433838552652-f9a46b332c40", + }, + { + img: "https://images.unsplash.com/photo-1506125840744-167167210587", + }, + { + img: "https://images.unsplash.com/photo-1522199873717-bc67b1a5e32b", + }, + { + img: "https://images.unsplash.com/photo-1495904786722-d2b5a19a8535", + }, + { + img: "https://images.unsplash.com/photo-1614094082869-cd4e4b2905c7", + }, + { + img: "https://images.unsplash.com/photo-1474755032398-4b0ed3b2ae5c", + }, + { + img: "https://images.unsplash.com/photo-1501554728187-ce583db33af7", + }, + { + img: "https://images.unsplash.com/photo-1515859005217-8a1f08870f59", + }, + { + img: "https://images.unsplash.com/photo-1531141445733-14c2eb7d4c1f", + }, + { + img: "https://images.unsplash.com/photo-1500259783852-0ca9ce8a64dc", + }, + { + img: "https://images.unsplash.com/photo-1510662145379-13537db782dc", + }, + { + img: "https://images.unsplash.com/photo-1573790387438-4da905039392", + }, + { + img: "https://images.unsplash.com/photo-1512757776214-26d36777b513", + }, + { + img: "https://images.unsplash.com/photo-1518855706573-84de4022b69b", + }, + { + img: "https://images.unsplash.com/photo-1500049242364-5f500807cdd7", + }, + { + img: "https://images.unsplash.com/photo-1528759335187-3b683174c86a", + }, +]; +export const THUMBNAIL_PARAMS = "w=240&h=240&fit=crop&auto=format"; + +// Console styles. +export const CONSOLE_BASE_STYLES = [ + "font-size: 12px", + "padding: 4px", + "border: 2px solid #5a5a5a", + "color: white", +].join(";"); +export const CONSOLE_PRIMARY = [ + CONSOLE_BASE_STYLES, + "background-color: #13315a", +].join(";"); +export const CONSOLE_SUCCESS = [ + CONSOLE_BASE_STYLES, + "background-color: #385a4e", +].join(";"); +export const CONSOLE_ERROR = [ + CONSOLE_BASE_STYLES, + "background-color: #5a1a24", +].join(";"); + +// Layouts. +export const LAYOUT_4_COLUMNS = { + name: "Layout 4 columns", + columns: 4, + itemWidth: 240, + itemHeight: 240, +}; +export const LAYOUT_8_COLUMNS = { + name: "Layout 8 columns", + columns: 8, + itemWidth: 240, + itemHeight: 240, +}; +export const LAYOUTS = [LAYOUT_4_COLUMNS, LAYOUT_8_COLUMNS]; + +export const createImageFile = async (src) => + new Promise((resolve, reject) => { + const img = new Image(); + img.src = src; + img.onload = () => resolve(img); + img.onerror = () => reject(new Error("Failed to construct image.")); + }); + +export const loadImage = async (url) => { + try { + const response = await fetch(url); + if (!response.ok) { + throw new Error(String(response.status)); + } + + return await response.blob(); + } catch (e) { + console.log(`%cFETCHED_FAILED: ${e}`, CONSOLE_ERROR); + } +}; + +export const weakRefCache = (fetchImg) => { + const imgCache = new Map(); + const registry = new FinalizationRegistry(({ imgName, size, type }) => { + const cachedImg = imgCache.get(imgName); + if (cachedImg && !cachedImg.deref()) { + imgCache.delete(imgName); + console.log( + `%cCLEANED_IMAGE: Url: ${imgName}, Size: ${size}, Type: ${type}`, + CONSOLE_ERROR, + ); + + const logEl = document.createElement("div"); + logEl.classList.add("logger-item", "logger--error"); + logEl.innerHTML = `CLEANED_IMAGE: Url: ${imgName}, Size: ${size}, Type: ${type}`; + loggerContainerEl.appendChild(logEl); + loggerContainerEl.scrollTop = loggerContainerEl.scrollHeight; + } + }); + + return async (imgName) => { + const cachedImg = imgCache.get(imgName); + + if (cachedImg?.deref() !== undefined) { + console.log( + `%cCACHED_IMAGE: Url: ${imgName}, Size: ${cachedImg.size}, Type: ${cachedImg.type}`, + CONSOLE_SUCCESS, + ); + + const logEl = document.createElement("div"); + logEl.classList.add("logger-item", "logger--success"); + logEl.innerHTML = `CACHED_IMAGE: Url: ${imgName}, Size: ${cachedImg.size}, Type: ${cachedImg.type}`; + loggerContainerEl.appendChild(logEl); + loggerContainerEl.scrollTop = loggerContainerEl.scrollHeight; + + return cachedImg?.deref(); + } + + const newImg = await fetchImg(imgName); + console.log( + `%cFETCHED_IMAGE: Url: ${imgName}, Size: ${newImg.size}, Type: ${newImg.type}`, + CONSOLE_PRIMARY, + ); + + const logEl = document.createElement("div"); + logEl.classList.add("logger-item", "logger--primary"); + logEl.innerHTML = `FETCHED_IMAGE: Url: ${imgName}, Size: ${newImg.size}, Type: ${newImg.type}`; + loggerContainerEl.appendChild(logEl); + loggerContainerEl.scrollTop = loggerContainerEl.scrollHeight; + + imgCache.set(imgName, new WeakRef(newImg)); + registry.register(newImg, { + imgName, + size: newImg.size, + type: newImg.type, + }); + + return newImg; + }; +}; + +export const stateObj = { + loading: false, + drawing: true, + collageRendered: false, + currentLayout: LAYOUTS[0], + selectedImages: new Set(), +}; diff --git a/1-js/99-js-misc/index.md b/1-js/99-js-misc/index.md new file mode 100644 index 000000000..79cd72fe7 --- /dev/null +++ b/1-js/99-js-misc/index.md @@ -0,0 +1,2 @@ + +# Miscellaneous diff --git a/10-misc/12-mutation-observer/article.md b/10-misc/12-mutation-observer/article.md deleted file mode 100644 index 1045e87a8..000000000 --- a/10-misc/12-mutation-observer/article.md +++ /dev/null @@ -1,249 +0,0 @@ - -# Mutation observer - -`MutationObserver` is a built-in object that observes a DOM element and fires a callback in case of changes. - -We'll first see syntax, and then explore a real-world use case. - -## Syntax - -`MutationObserver` is easy to use. - -First, we create an observer with a callback-function: - -```js -let observer = new MutationObserver(callback); -``` - -And then attach it to a DOM node: - -```js -observer.observe(node, config); -``` - -`config` is an object with boolean options "what kind of changes to react on": -- `childList` -- changes in the direct children of `node`, -- `subtree` -- in all descendants of `node`, -- `attributes` -- attributes of `node`, -- `attributeOldValue` -- record the old value of attribute (infers `attributes`), -- `characterData` -- whether to observe `node.data` (text content), -- `characterDataOldValue` -- record the old value of `node.data` (infers `characterData`), -- `attributeFilter` -- an array of attribute names, to observe only selected ones. - -Then after any changes, the `callback` is executed, with a list of [MutationRecord](https://dom.spec.whatwg.org/#mutationrecord) objects as the first argument, and the observer itself as the second argument. - -[MutationRecord](https://dom.spec.whatwg.org/#mutationrecord) objects have properties: - -- `type` -- mutation type, one of - - `"attributes"` (attribute modified) - - `"characterData"` (data modified) - - `"childList"` (elements added/removed), -- `target` -- where the change occured: an element for "attributes", or text node for "characterData", or an element for a "childList" mutation, -- `addedNodes/removedNodes` -- nodes that were added/removed, -- `previousSibling/nextSibling` -- the previous and next sibling to added/removed nodes, -- `attributeName/attributeNamespace` -- the name/namespace (for XML) of the changed attribute, -- `oldValue` -- the previous value, only for attribute or text changes. - - -For example, here's a `<div>` with `contentEditable` attribute. That attribute allows us to focus on it and edit. - -```html run -<div contentEditable id="elem">Edit <b>me</b>, please</div> - -<script> -let observer = new MutationObserver(mutationRecords => { - console.log(mutationRecords); // console.log(the changes) -}); -observer.observe(elem, { - // observe everything except attributes - childList: true, - subtree: true, - characterDataOldValue: true -}); -</script> -``` - -If we change the text inside `<b>me</b>`, we'll get a single mutation: - -```js -mutationRecords = [{ - type: "characterData", - oldValue: "me", - target: <text node>, - // other properties empty -}]; -``` - -If we select and remove the `<b>me</b>` altogether, we'll get multiple mutations: - -```js -mutationRecords = [{ - type: "childList", - target: <div#elem>, - removedNodes: [<b>], - nextSibling: <text node>, - previousSibling: <text node> - // other properties empty -}, { - type: "characterData" - target: <text node> - // ...details depend on how the browser handles the change - // it may coalesce two adjacent text nodes "Edit " and ", please" into one node - // or it can just delete the extra space after "Edit". - // may be one mutation or a few -}]; -``` - -## Observer use case - -When `MutationObserver` is needed? Is there a scenario when such thing can be useful? - -Sure, we can track something like `contentEditable` and create "undo/redo" stack, but here's an example where `MutationObserver` is good from architectural standpoint. - -Let's say we're making a website about programming, like this one. Naturally, articles and other materials may contain source code snippets. - -An HTML code snippet looks like this: -```html -... -<pre class="language-javascript"><code> - // here's the code - let hello = "world"; -</code></pre> -... -``` - -There's also a JavaScript highlighting library, e.g. [Prism.js](https://prismjs.com/). A call to `Prism.highlightElem(pre)` examines the contents of such `pre` elements and adds colored syntax highlighting, similar to what you in examples here, this page. - -Generally, when a page loads, e.g. at the bottom of the page, we can search for elements `pre[class*="language"]` and call `Prism.highlightElem` on them: - -```js -// highlight all code snippets on the page -document.querySelectorAll('pre[class*="language"]').forEach(Prism.highlightElem); -``` - -Now the `<pre>` snippet looks like this (without line numbers by default): - -```js -// here's the code -let hello = "world"; -``` - -Everything's simple so far, right? There are `<pre>` code snippets in HTML, we highlight them. - -Now let's go on. Let's say we're going to dynamically fetch materials from a server. We'll study methods for that [later in the tutorial](info:fetch-basics). For now it only matters that we fetch an HTML article from a webserver and display it on demand: - -```js -let article = /* fetch new content from server */ -articleElem.innerHTML = article; -``` - -The new `article` HTML may contain code snippets. We need to call `Prism.highlightElem` on them, otherwise they won't get highlighted. - -**Who's responsibility is to call `Prism.highlightElem` for a dynamically loaded article?** - -We could append that call to the code that loads an article, like this: - -```js -let article = /* fetch new content from server */ -articleElem.innerHTML = article; - -*!* -let snippets = articleElem.querySelectorAll('pre[class*="language-"]'); -snippets.forEach(Prism.highlightElem); -*/!* -``` - -...But imagine, we have many places where we load contents with code: articles, quizzes, forum posts. Do we need to put the highlighting call everywhere? Then we need to be careful, not to forget about it. - -And what if we load the content into a third-party engine? E.g. we have a forum written by someone else, that loads contents dynamically, and we'd like to add syntax highlighting to it. No one likes to patch third-party scripts. - -Luckily, there's another option. - -We can use `MutationObserver` to automatically detect code snippets inserted in the page and highlight them. - -So we'll handle the highlighting functionality in one place, relieving us from the need to integrate it. - -## Dynamic highlight demo - -Here's the working example. - -If you run this code, it starts observing the element below and highlighting any code snippets that appear there: - -```js run -let observer = new MutationObserver(mutations => { - - for(let mutation of mutations) { - // examine new nodes - - for(let node of mutation.addedNodes) { - // skip newly added text nodes - if (!(node instanceof HTMLElement)) continue; - - // check the inserted element for being a code snippet - if (node.matches('pre[class*="language-"]')) { - Prism.highlightElement(node); - } - - // search its subtree for code snippets - for(let elem of node.querySelectorAll('pre[class*="language-"]')) { - Prism.highlightElement(elem); - } - } - } - -}); - -let demoElem = document.getElementById('highlight-demo'); - -observer.observe(demoElem, {childList: true, subtree: true}); -``` - -<p id="highlight-demo" style="border: 1px solid #ddd">Demo element with <code>id="highlight-demo"</code>, obverved by the example above.</p> - -The code below populates `innerHTML`. If you've run the code above, snippets will get highlighted: - -```js run -let demoElem = document.getElementById('highlight-demo'); - -// dynamically insert content with code snippets -demoElem.innerHTML = `A code snippet is below: - <pre class="language-javascript"><code> let hello = "world!"; </code></pre> - <div>Another one:</div> - <div> - <pre class="language-css"><code>.class { margin: 5px; } </code></pre> - </div> -`; -``` - -Now we have `MutationObserver` that can track all highlighting in observed elements or the whole `document`. We can add/remove code snippets in HTML without thinking about it. - - -## Garbage collection - -Observers use weak references to nodes internally. That is: if a node is removed from DOM, and becomes unreachable, then it becomes garbage collected, an observer doesn't prevent that. - -Still, we can release observers any time: - -- `observer.disconnect()` -- stops the observation. - -Additionally: - -- `mutationRecords = observer.takeRecords()` -- gets a list of unprocessed mutation records, those that happened, but the callback did not handle them. - -```js -// we're going to disconnect the observer -// it might have not yet handled some mutations -let mutationRecords = observer.takeRecords(); -// process mutationRecords - -// now all handled, disconnect -observer.disconnect(); -``` - -## Summary - -`MutationObserver` can react on changes in DOM: attributes, added/removed elements, text content. - -We can use it to track changes introduced by other parts of our own or 3rd-party code. - -For example, to post-process dynamically inserted content, as demo `innerHTML`, like highlighting in the example above. diff --git a/10-misc/index.md b/10-misc/index.md deleted file mode 100644 index 65ab3188a..000000000 --- a/10-misc/index.md +++ /dev/null @@ -1,4 +0,0 @@ - -# Miscellaneous - -Not yet categorized articles. diff --git a/2-ui/1-document/01-browser-environment/article.md b/2-ui/1-document/01-browser-environment/article.md index 0e123f581..eedc28fb3 100644 --- a/2-ui/1-document/01-browser-environment/article.md +++ b/2-ui/1-document/01-browser-environment/article.md @@ -1,12 +1,12 @@ # Browser environment, specs -The JavaScript language was initially created for web browsers. Since then, it has evolved and become a language with many uses and platforms. +The JavaScript language was initially created for web browsers. Since then, it has evolved into a language with many uses and platforms. -A platform may be a browser, or a web-server, or a washing machine, or another *host*. Each of them provides platform-specific functionality. The JavaScript specification calls that a *host environment*. +A platform may be a browser, or a web-server or another *host*, or even a "smart" coffee machine if it can run JavaScript. Each of these provides platform-specific functionality. The JavaScript specification calls that a *host environment*. -A host environment provides platform-specific objects and functions additional to the language core. Web browsers give a means to control web pages. Node.js provides server-side features, and so on. +A host environment provides its own objects and functions in addition to the language core. Web browsers give a means to control web pages. Node.js provides server-side features, and so on. -Here's a bird's-eye view of what we have when JavaScript runs in a web-browser: +Here's a bird's-eye view of what we have when JavaScript runs in a web browser:  @@ -15,28 +15,30 @@ There's a "root" object called `window`. It has two roles: 1. First, it is a global object for JavaScript code, as described in the chapter <info:global-object>. 2. Second, it represents the "browser window" and provides methods to control it. -For instance, here we use it as a global object: +For instance, we can use it as a global object: -```js run +```js run global function sayHi() { alert("Hello"); } -// global functions are accessible as properties of window +// global functions are methods of the global object: window.sayHi(); ``` -And here we use it as a browser window, to see the window height: +And we can use it as a browser window, to show the window height: ```js run alert(window.innerHeight); // inner window height ``` -There are more window-specific methods and properties, we'll cover them later. +There are more window-specific methods and properties, which we'll cover later. + +## DOM (Document Object Model) -## Document Object Model (DOM) +The Document Object Model, or DOM for short, represents all page content as objects that can be modified. -The `document` object gives access to the page content. We can change or create anything on the page using it. +The `document` object is the main "entry point" to the page. We can change or create anything on the page using it. For instance: ```js run @@ -47,58 +49,44 @@ document.body.style.background = "red"; setTimeout(() => document.body.style.background = "", 1000); ``` -Here we used `document.body.style`, but there's much, much more. Properties and methods are described in the specification. There happen to be two working groups who develop it: - -1. [W3C](https://en.wikipedia.org/wiki/World_Wide_Web_Consortium) -- the documentation is at <https://www.w3.org/TR/dom>. -2. [WhatWG](https://en.wikipedia.org/wiki/WHATWG), publishing at <https://dom.spec.whatwg.org>. - -As it happens, the two groups don't always agree, so it's like we have two sets of standards. But they are very similar and eventually things merge. The documentation that you can find on the given resources is very similar, with about a 99% match. There are very minor differences that you probably won't notice. - -Personally, I find <https://dom.spec.whatwg.org> more pleasant to use. - -In the ancient past, there was no standard at all -- each browser implemented however it wanted. Different browsers had different sets, methods, and properties for the same thing, and developers had to write different code for each of them. Dark, messy times. - -Even now we can sometimes meet old code that uses browser-specific properties and works around incompatibilities. But, in this tutorial we'll use modern stuff: there's no need to learn old things until you really need to (chances are high that you won't). - -Then the DOM standard appeared, in an attempt to bring everyone to an agreement. The first version was "DOM Level 1", then it was extended by DOM Level 2, then DOM Level 3, and now it's reached DOM Level 4. People from WhatWG group got tired of version numbers and are calling it just "DOM", without a number. So we'll do the same. +Here, we used `document.body.style`, but there's much, much more. Properties and methods are described in the specification: [DOM Living Standard](https://dom.spec.whatwg.org). ```smart header="DOM is not only for browsers" -The DOM specification explains the structure of a document and provides objects to manipulate it. There are non-browser instruments that use it too. +The DOM specification explains the structure of a document and provides objects to manipulate it. There are non-browser instruments that use DOM too. -For instance, server-side tools that download HTML pages and process them use the DOM. They may support only a part of the specification though. +For instance, server-side scripts that download HTML pages and process them can also use the DOM. They may support only a part of the specification though. ``` ```smart header="CSSOM for styling" -CSS rules and stylesheets are not structured like HTML. There's a separate specification [CSSOM](https://www.w3.org/TR/cssom-1/) that explains how they are represented as objects, and how to read and write them. +There's also a separate specification, [CSS Object Model (CSSOM)](https://www.w3.org/TR/cssom-1/) for CSS rules and stylesheets, that explains how they are represented as objects, and how to read and write them. -CSSOM is used together with DOM when we modify style rules for the document. In practice though, CSSOM is rarely required, because usually CSS rules are static. We rarely need to add/remove CSS rules from JavaScript, so we won't cover it right now. +The CSSOM is used together with the DOM when we modify style rules for the document. In practice though, the CSSOM is rarely required, because we rarely need to modify CSS rules from JavaScript (usually we just add/remove CSS classes, not modify their CSS rules), but that's also possible. ``` -## BOM (part of HTML spec) +## BOM (Browser Object Model) -Browser Object Model (BOM) are additional objects provided by the browser (host environment) to work with everything except the document. +The Browser Object Model (BOM) represents additional objects provided by the browser (host environment) for working with everything except the document. For instance: -- The [navigator](mdn:api/Window/navigator) object provides background information about the browser and the operating system. There are many properties, but the two most widely known are: `navigator.userAgent` -- about the current browser, and `navigator.platform` -- about the platform (can help to differ between Windows/Linux/Mac etc). +- The [navigator](mdn:api/Window/navigator) object provides background information about the browser and the operating system. There are many properties, but the two most widely known are: `navigator.userAgent` -- about the current browser, and `navigator.platform` -- about the platform (can help to differentiate between Windows/Linux/Mac etc). - The [location](mdn:api/Window/location) object allows us to read the current URL and can redirect the browser to a new one. Here's how we can use the `location` object: ```js run alert(location.href); // shows current URL -if (confirm("Go to wikipedia?")) { +if (confirm("Go to Wikipedia?")) { location.href = "https://wikipedia.org"; // redirect the browser to another URL } ``` -Functions `alert/confirm/prompt` are also a part of BOM: they are directly not related to the document, but represent pure browser methods of communicating with the user. - +The functions `alert/confirm/prompt` are also a part of the BOM: they are not directly related to the document, but represent pure browser methods for communicating with the user. -```smart header="HTML specification" -BOM is the part of the general [HTML specification](https://html.spec.whatwg.org). +```smart header="Specifications" +The BOM is a part of the general [HTML specification](https://html.spec.whatwg.org). -Yes, you heard that right. The HTML spec at <https://html.spec.whatwg.org> is not only about the "HTML language" (tags, attributes), but also covers a bunch of objects, methods and browser-specific DOM extensions. That's "HTML in broad terms". +Yes, you heard that right. The HTML spec at <https://html.spec.whatwg.org> is not only about the "HTML language" (tags, attributes), but also covers a bunch of objects, methods, and browser-specific DOM extensions. That's "HTML in broad terms". Also, some parts have additional specs listed at <https://spec.whatwg.org>. ``` ## Summary @@ -106,16 +94,20 @@ Yes, you heard that right. The HTML spec at <https://html.spec.whatwg.org> is no Talking about standards, we have: DOM specification -: Describes the document structure, manipulations and events, see <https://dom.spec.whatwg.org>. +: Describes the document structure, manipulations, and events, see <https://dom.spec.whatwg.org>. CSSOM specification -: Describes stylesheets and style rules, manipulations with them and their binding to documents, see <https://www.w3.org/TR/cssom-1/>. +: Describes stylesheets and style rules, manipulations with them, and their binding to documents, see <https://www.w3.org/TR/cssom-1/>. HTML specification : Describes the HTML language (e.g. tags) and also the BOM (browser object model) -- various browser functions: `setTimeout`, `alert`, `location` and so on, see <https://html.spec.whatwg.org>. It takes the DOM specification and extends it with many additional properties and methods. -Now we'll get down to learning DOM, because the document plays the central role in the UI. +Additionally, some classes are described separately at <https://spec.whatwg.org/>. + +Please note these links, as there's so much to learn that it's impossible to cover everything and remember it all. + +When you'd like to read about a property or a method, the Mozilla manual at <https://developer.mozilla.org/en-US/> is also a nice resource, but the corresponding spec may be better: it's more complex and longer to read, but will make your fundamental knowledge sound and complete. -Please note the links above, as there's so much stuff to learn it's impossible to cover and remember everything. +To find something, it's often convenient to use an internet search "WHATWG [term]" or "MDN [term]", e.g <https://google.com?q=whatwg+localstorage>, <https://google.com?q=mdn+localstorage>. -When you'd like to read about a property or a method, the Mozilla manual at <https://developer.mozilla.org/en-US/search> is a nice resource, but reading the corresponding spec may be better: it's more complex and longer to read, but will make your fundamental knowledge sound and complete. +Now, we'll get down to learning the DOM, because the document plays the central role in the UI. diff --git a/2-ui/1-document/02-dom-nodes/article.md b/2-ui/1-document/02-dom-nodes/article.md index 6bb34ff8c..f7f2be91d 100644 --- a/2-ui/1-document/02-dom-nodes/article.md +++ b/2-ui/1-document/02-dom-nodes/article.md @@ -6,26 +6,42 @@ libs: # DOM tree -The backbone of an HTML document are tags. +The backbone of an HTML document is tags. -According to Document Object Model (DOM), every HTML-tag is an object. Nested tags are called "children" of the enclosing one. +According to the Document Object Model (DOM), every HTML tag is an object. Nested tags are "children" of the enclosing one. The text inside a tag is an object as well. -The text inside a tag it is an object as well. +All these objects are accessible using JavaScript, and we can use them to modify the page. -All these objects are accessible using JavaScript. +For example, `document.body` is the object representing the `<body>` tag. -## An example of DOM +Running this code will make the `<body>` red for 3 seconds: -For instance, let's explore the DOM for this document: +```js run +document.body.style.background = 'red'; // make the background red + +setTimeout(() => document.body.style.background = '', 3000); // return back +``` + +Here we used `style.background` to change the background color of `document.body`, but there are many other properties, such as: + +- `innerHTML` -- HTML contents of the node. +- `offsetWidth` -- the node width (in pixels) +- ...and so on. + +Soon we'll learn more ways to manipulate the DOM, but first we need to know about its structure. + +## An example of the DOM + +Let's start with the following simple document: ```html run no-beautify <!DOCTYPE HTML> <html> <head> - <title>About elks</title> + <title>About elk</title> </head> <body> - The truth about elks. + The truth about elk. </body> </html> ``` @@ -35,7 +51,7 @@ The DOM represents HTML as a tree structure of tags. Here's how it looks: <div class="domtree"></div> <script> -let node1 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n "},{"name":"TITLE","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"About elks"}]},{"name":"#text","nodeType":3,"content":"\n "}]},{"name":"#text","nodeType":3,"content":"\n "},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n The truth about elks."}]}]} +let node1 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n "},{"name":"TITLE","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"About elk"}]},{"name":"#text","nodeType":3,"content":"\n"}]},{"name":"#text","nodeType":3,"content":"\n"},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n The truth about elk.\n"}]}]} drawHtmlTree(node1, 'div.domtree', 690, 320); </script> @@ -44,56 +60,57 @@ drawHtmlTree(node1, 'div.domtree', 690, 320); On the picture above, you can click on element nodes and their children will open/collapse. ``` -Tags are called *element nodes* (or just elements). Nested tags become children of the enclosing ones. As a result we have a tree of elements: `<html>` is at the root, then `<head>` and `<body>` are its children, etc. +Every tree node is an object. + +Tags are *element nodes* (or just elements) and form the tree structure: `<html>` is at the root, then `<head>` and `<body>` are its children, etc. The text inside elements forms *text nodes*, labelled as `#text`. A text node contains only a string. It may not have children and is always a leaf of the tree. -For instance, the `<title>` tag has the text `"About elks"`. +For instance, the `<title>` tag has the text `"About elk"`. Please note the special characters in text nodes: - a newline: `↵` (in JavaScript known as `\n`) - a space: `␣` -Spaces and newlines -- are totally valid characters, they form text nodes and become a part of the DOM. So, for instance, in the example above the `<head>` tag contains some spaces before `<title>`, and that text becomes a `#text` node (it contains a newline and some spaces only). +Spaces and newlines are totally valid characters, like letters and digits. They form text nodes and become a part of the DOM. So, for instance, in the example above the `<head>` tag contains some spaces before `<title>`, and that text becomes a `#text` node (it contains a newline and some spaces only). There are only two top-level exclusions: -1. Spaces and newlines before `<head>` are ignored for historical reasons, -2. If we put something after `</body>`, then that is automatically moved inside the `body`, at the end, as the HTML spec requires that all content must be inside `<body>`. So there may be no spaces after `</body>`. +1. Spaces and newlines before `<head>` are ignored for historical reasons. +2. If we put something after `</body>`, then that is automatically moved inside the `body`, at the end, as the HTML spec requires that all content must be inside `<body>`. So there can't be any spaces after `</body>`. -In other cases everything's straightforward -- if there are spaces (just like any character) in the document, then they become text nodes in DOM, and if we remove them, then there won't be any. +In other cases everything's straightforward -- if there are spaces (just like any character) in the document, then they become text nodes in the DOM, and if we remove them, then there won't be any. Here are no space-only text nodes: ```html no-beautify <!DOCTYPE HTML> -<html><head><title>About elks</title></head><body>The truth about elks.</body></html> +<html><head><title>About elk</title></head><body>The truth about elk.</body></html> ``` <div class="domtree"></div> <script> -let node2 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[{"name":"TITLE","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"About elks"}]}]},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"The truth about elks."}]}]} +let node2 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[{"name":"TITLE","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"About elk"}]}]},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"The truth about elk."}]}]} drawHtmlTree(node2, 'div.domtree', 690, 210); </script> -```smart header="Edge spaces and in-between empty text are usually hidden in tools" +```smart header="Spaces at string start/end and space-only text nodes are usually hidden in tools" Browser tools (to be covered soon) that work with DOM usually do not show spaces at the start/end of the text and empty text nodes (line-breaks) between tags. -That's because they are mainly used to decorate HTML, and do not affect how it is shown (in most cases). +Developer tools save screen space this way. -On further DOM pictures we'll sometimes omit them where they are irrelevant, to keep things short. +On further DOM pictures we'll sometimes omit them when they are irrelevant. Such spaces usually do not affect how the document is displayed. ``` - ## Autocorrection -If the browser encounters malformed HTML, it automatically corrects it when making DOM. +If the browser encounters malformed HTML, it automatically corrects it when making the DOM. -For instance, the top tag is always `<html>`. Even if it doesn't exist in the document -- it will exist in the DOM, the browser will create it. The same goes for `<body>`. +For instance, the top tag is always `<html>`. Even if it doesn't exist in the document, it will exist in the DOM, because the browser will create it. The same goes for `<body>`. -As an example, if the HTML file is a single word `"Hello"`, the browser will wrap it into `<html>` and `<body>`, add the required `<head>`, and the DOM will be: +As an example, if the HTML file is the single word `"Hello"`, the browser will wrap it into `<html>` and `<body>`, and add the required `<head>`, and the DOM will be: <div class="domtree"></div> @@ -106,7 +123,7 @@ drawHtmlTree(node3, 'div.domtree', 690, 150); While generating the DOM, browsers automatically process errors in the document, close tags and so on. -Such an "invalid" document: +A document with unclosed tags: ```html no-beautify <p>Hello @@ -115,7 +132,7 @@ Such an "invalid" document: <li>Dad ``` -...Will become a normal DOM, as the browser reads tags and restores the missing parts: +...will become a normal DOM as the browser reads tags and restores the missing parts: <div class="domtree"></div> @@ -126,7 +143,7 @@ drawHtmlTree(node4, 'div.domtree', 690, 360); </script> ````warn header="Tables always have `<tbody>`" -An interesting "special case" is tables. By the DOM specification they must have `<tbody>`, but HTML text may (officially) omit it. Then the browser creates `<tbody>` in DOM automatically. +An interesting "special case" is tables. By DOM specification they must have `<tbody>` tag, but HTML text may omit it. Then the browser creates `<tbody>` in the DOM automatically. For the HTML: @@ -143,18 +160,20 @@ let node5 = {"name":"TABLE","nodeType":1,"children":[{"name":"TBODY","nodeType": drawHtmlTree(node5, 'div.domtree', 600, 200); </script> -You see? The `<tbody>` appeared out of nowhere. You should keep this in mind while working with tables to avoid surprises. +You see? The `<tbody>` appeared out of nowhere. We should keep this in mind while working with tables to avoid surprises. ```` ## Other node types -Let's add more tags and a comment to the page: +There are some other node types besides elements and text nodes. + +For example, comments: ```html <!DOCTYPE HTML> <html> <body> - The truth about elks. + The truth about elk. <ol> <li>An elk is a smart</li> *!* @@ -169,18 +188,18 @@ Let's add more tags and a comment to the page: <div class="domtree"></div> <script> -let node6 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[]},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n The truth about elks.\n "},{"name":"OL","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n "},{"name":"LI","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"An elk is a smart"}]},{"name":"#text","nodeType":3,"content":"\n "},{"name":"#comment","nodeType":8,"content":"comment"},{"name":"#text","nodeType":3,"content":"\n "},{"name":"LI","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"...and cunning animal!"}]},{"name":"#text","nodeType":3,"content":"\n "}]},{"name":"#text","nodeType":3,"content":"\n \n"}]}]}; +let node6 = {"name":"HTML","nodeType":1,"children":[{"name":"HEAD","nodeType":1,"children":[]},{"name":"BODY","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n The truth about elk.\n "},{"name":"OL","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"\n "},{"name":"LI","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"An elk is a smart"}]},{"name":"#text","nodeType":3,"content":"\n "},{"name":"#comment","nodeType":8,"content":"comment"},{"name":"#text","nodeType":3,"content":"\n "},{"name":"LI","nodeType":1,"children":[{"name":"#text","nodeType":3,"content":"...and cunning animal!"}]},{"name":"#text","nodeType":3,"content":"\n "}]},{"name":"#text","nodeType":3,"content":"\n\n\n"}]}]}; drawHtmlTree(node6, 'div.domtree', 690, 500); </script> -Here we see a new tree node type -- *comment node*, labeled as `#comment`. +We can see here a new tree node type -- *comment node*, labeled as `#comment`, between two text nodes. We may think -- why is a comment added to the DOM? It doesn't affect the visual representation in any way. But there's a rule -- if something's in HTML, then it also must be in the DOM tree. **Everything in HTML, even comments, becomes a part of the DOM.** -Even the `<!DOCTYPE...>` directive at the very beginning of HTML is also a DOM node. It's in the DOM tree right before `<html>`. We are not going to touch that node, we even don't draw it on diagrams for that reason, but it's there. +Even the `<!DOCTYPE...>` directive at the very beginning of HTML is also a DOM node. It's in the DOM tree right before `<html>`. Few people know about that. We are not going to touch that node, we even don't draw it on diagrams, but it's there. The `document` object that represents the whole document is, formally, a DOM node as well. @@ -189,31 +208,29 @@ There are [12 node types](https://dom.spec.whatwg.org/#node). In practice we usu 1. `document` -- the "entry point" into DOM. 2. element nodes -- HTML-tags, the tree building blocks. 3. text nodes -- contain text. -4. comments -- sometimes we can put the information there, it won't be shown, but JS can read it from the DOM. +4. comments -- sometimes we can put information there, it won't be shown, but JS can read it from the DOM. ## See it for yourself -To see the DOM structure in real-time, try [Live DOM Viewer](http://software.hixie.ch/utilities/js/live-dom-viewer/). Just type in the document, and it will show up DOM at an instant. - -## In the browser inspector +To see the DOM structure in real-time, try [Live DOM Viewer](https://software.hixie.ch/utilities/js/live-dom-viewer/). Just type in the document, and it will show up as a DOM at an instant. Another way to explore the DOM is to use the browser developer tools. Actually, that's what we use when developing. -To do so, open the web-page [elks.html](elks.html), turn on the browser developer tools and switch to the Elements tab. +To do so, open the web page [elk.html](elk.html), turn on the browser developer tools and switch to the Elements tab. It should look like this: - + You can see the DOM, click on elements, see their details and so on. Please note that the DOM structure in developer tools is simplified. Text nodes are shown just as text. And there are no "blank" (space only) text nodes at all. That's fine, because most of the time we are interested in element nodes. -Clicking the <span class="devtools" style="background-position:-328px -124px"></span> button in the left-upper corner allows to choose a node from the webpage using a mouse (or other pointer devices) and "inspect" it (scroll to it in the Elements tab). This works great when we have a huge HTML page (and corresponding huge DOM) and would like to see the place of a particular element in it. +Clicking the <span class="devtools" style="background-position:-328px -124px"></span> button in the left-upper corner allows us to choose a node from the webpage using a mouse (or other pointer devices) and "inspect" it (scroll to it in the Elements tab). This works great when we have a huge HTML page (and corresponding huge DOM) and would like to see the place of a particular element in it. Another way to do it would be just right-clicking on a webpage and selecting "Inspect" in the context menu. - + At the right part of the tools there are the following subtabs: - **Styles** -- we can see CSS applied to the current element rule by rule, including built-in rules (gray). Almost everything can be edited in-place, including the dimensions/margins/paddings of the box below. @@ -225,22 +242,26 @@ The best way to study them is to click around. Most values are editable in-place ## Interaction with console -As we explore the DOM, we also may want to apply JavaScript to it. Like: get a node and run some code to modify it, to see how it looks. Here are few tips to travel between the Elements tab and the console. +As we work the DOM, we also may want to apply JavaScript to it. Like: get a node and run some code to modify it, to see the result. Here are few tips to travel between the Elements tab and the console. + +For the start: -- Select the first `<li>` in the Elements tab. -- Press `key:Esc` -- it will open console right below the Elements tab. +1. Select the first `<li>` in the Elements tab. +2. Press `key:Esc` -- it will open console right below the Elements tab. Now the last selected element is available as `$0`, the previously selected is `$1` etc. We can run commands on them. For instance, `$0.style.background = 'red'` makes the selected list item red, like this: - + + +That's how to get a node from Elements in Console. -From the other side, if we're in console and have a variable referencing a DOM node, then we can use the command `inspect(node)` to see it in the Elements pane. +There's also a road back. If there's a variable referencing a DOM node, then we can use the command `inspect(node)` in Console to see it in the Elements pane. -Or we can just output it in the console and explore "at-place", like `document.body` below: +Or we can just output the DOM node in the console and explore "in-place", like `document.body` below: - + That's for debugging purposes of course. From the next chapter on we'll access and modify DOM using JavaScript. @@ -258,4 +279,4 @@ We can use developer tools to inspect DOM and modify it manually. Here we covered the basics, the most used and important actions to start with. There's an extensive documentation about Chrome Developer Tools at <https://developers.google.com/web/tools/chrome-devtools>. The best way to learn the tools is to click here and there, read menus: most options are obvious. Later, when you know them in general, read the docs and pick up the rest. -DOM nodes have properties and methods that allow to travel between them, modify, move around the page and more. We'll get down to them in the next chapters. +DOM nodes have properties and methods that allow us to travel between them, modify them, move around the page, and more. We'll get down to them in the next chapters. diff --git a/2-ui/1-document/02-dom-nodes/domconsole0.png b/2-ui/1-document/02-dom-nodes/domconsole0.png deleted file mode 100644 index 121c11d75..000000000 Binary files a/2-ui/1-document/02-dom-nodes/domconsole0.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/domconsole0.svg b/2-ui/1-document/02-dom-nodes/domconsole0.svg new file mode 100644 index 000000000..eb99f193f --- /dev/null +++ b/2-ui/1-document/02-dom-nodes/domconsole0.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="699" height="450" viewBox="0 0 699 450"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h699v450H0z"/><path id="path-2" d="M72 305v21.429l5.25-5.358L81.625 330h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H86L72 305z"/><filter id="filter-3" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="domconsole0.svg"><g id="Bitmap"><image width="699" height="450" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><path id="Rectangle" stroke="#C06334" stroke-width="2" d="M4 393h209v53H4z"/><g id="Default"><use fill="#000" filter="url(#filter-3)" xlink:href="#path-2"/><path fill="#FFF" stroke="#A7333A" d="M71.5 303.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727v-23.878z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/02-dom-nodes/domconsole0@2x.png b/2-ui/1-document/02-dom-nodes/domconsole0@2x.png deleted file mode 100644 index a8953395c..000000000 Binary files a/2-ui/1-document/02-dom-nodes/domconsole0@2x.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/domconsole1.png b/2-ui/1-document/02-dom-nodes/domconsole1.png deleted file mode 100644 index c04f015cf..000000000 Binary files a/2-ui/1-document/02-dom-nodes/domconsole1.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/domconsole1.svg b/2-ui/1-document/02-dom-nodes/domconsole1.svg new file mode 100644 index 000000000..02ef5f0a6 --- /dev/null +++ b/2-ui/1-document/02-dom-nodes/domconsole1.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="696" height="512" viewBox="0 0 696 512"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h696v512H0z"/><path id="path-2" d="M66 424v21.429l5.25-5.358L75.625 449h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H80L66 424z"/><filter id="filter-3" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="domconsole1.svg"><g id="Bitmap"><image width="696" height="512" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="Default"><use fill="#000" filter="url(#filter-3)" xlink:href="#path-2"/><path fill="#FFF" stroke="#7E7C7B" d="M65.5 422.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727v-23.878z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/02-dom-nodes/domconsole1@2x.png b/2-ui/1-document/02-dom-nodes/domconsole1@2x.png deleted file mode 100644 index ce0fa0fff..000000000 Binary files a/2-ui/1-document/02-dom-nodes/domconsole1@2x.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/elks.html b/2-ui/1-document/02-dom-nodes/elk.html similarity index 86% rename from 2-ui/1-document/02-dom-nodes/elks.html rename to 2-ui/1-document/02-dom-nodes/elk.html index 7d29f3d4e..dc5d65f54 100644 --- a/2-ui/1-document/02-dom-nodes/elks.html +++ b/2-ui/1-document/02-dom-nodes/elk.html @@ -1,7 +1,7 @@ <!DOCTYPE HTML> <html> <body> - The truth about elks. + The truth about elk. <ol> <li>An elk is a smart</li> <!-- comment --> diff --git a/2-ui/1-document/02-dom-nodes/elk.svg b/2-ui/1-document/02-dom-nodes/elk.svg new file mode 100644 index 000000000..448eea9d1 --- /dev/null +++ b/2-ui/1-document/02-dom-nodes/elk.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="698" height="249" viewBox="0 0 698 249"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h698v249H0z"/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="Bitmap"><image width="698" height="249" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/02-dom-nodes/elks.png b/2-ui/1-document/02-dom-nodes/elks.png deleted file mode 100644 index 03177c40e..000000000 Binary files a/2-ui/1-document/02-dom-nodes/elks.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/elks@2x.png b/2-ui/1-document/02-dom-nodes/elks@2x.png deleted file mode 100644 index e8a15bd5b..000000000 Binary files a/2-ui/1-document/02-dom-nodes/elks@2x.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/inspect.png b/2-ui/1-document/02-dom-nodes/inspect.png deleted file mode 100644 index 075cf9308..000000000 Binary files a/2-ui/1-document/02-dom-nodes/inspect.png and /dev/null differ diff --git a/2-ui/1-document/02-dom-nodes/inspect.svg b/2-ui/1-document/02-dom-nodes/inspect.svg new file mode 100644 index 000000000..60696ec0d --- /dev/null +++ b/2-ui/1-document/02-dom-nodes/inspect.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="696" height="379" viewBox="0 0 696 379"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M0 0h696v379H0z"/><path id="path-2" d="M142 52v21.429l5.25-5.358L151.625 77h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H156L142 52z"/><filter id="filter-3" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="inspect.svg"><g id="Bitmap"><image width="696" height="379" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><g id="Default"><use fill="#000" filter="url(#filter-3)" xlink:href="#path-2"/><path fill="#FFF" stroke="#A7333A" d="M141.5 50.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727V50.775z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/02-dom-nodes/inspect@2x.png b/2-ui/1-document/02-dom-nodes/inspect@2x.png deleted file mode 100644 index 8743dd297..000000000 Binary files a/2-ui/1-document/02-dom-nodes/inspect@2x.png and /dev/null differ diff --git a/2-ui/1-document/03-dom-navigation/1-dom-children/task.md b/2-ui/1-document/03-dom-navigation/1-dom-children/task.md index 4a9e741a9..d97f2748a 100644 --- a/2-ui/1-document/03-dom-navigation/1-dom-children/task.md +++ b/2-ui/1-document/03-dom-navigation/1-dom-children/task.md @@ -4,7 +4,7 @@ importance: 5 # DOM children -For the page: +Look at this page: ```html <html> @@ -18,7 +18,7 @@ For the page: </html> ``` -How to access: +For each of the following, give at least one way of how to access them: - The `<div>` DOM node? - The `<ul>` DOM node? - The second `<li>` (with Pete)? diff --git a/2-ui/1-document/03-dom-navigation/3-navigation-links-which-null/solution.md b/2-ui/1-document/03-dom-navigation/3-navigation-links-which-null/solution.md index e962eb662..d76936320 100644 --- a/2-ui/1-document/03-dom-navigation/3-navigation-links-which-null/solution.md +++ b/2-ui/1-document/03-dom-navigation/3-navigation-links-which-null/solution.md @@ -1,4 +1,6 @@ -1. Yes, true. The element `elem.lastChild` is always the last one, it has no `nextSibling`, so if there are children, then yes. -2. No, wrong, because `elem.children[0]` is the first child among elements. But there may be non-element nodes before it. So `previousSibling` may be a text node. +1. Yes, true. The element `elem.lastChild` is always the last one, it has no `nextSibling`. +2. No, wrong, because `elem.children[0]` is the first child *among elements*. But there may exist non-element nodes before it. So `previousSibling` may be a text node. -Please note that for both cases if there are no children, then there will be an error. For instance, if `elem.lastChild` is `null`, we can't access `elem.lastChild.nextSibling`. +Please note: for both cases if there are no children, then there will be an error. + +If there are no children, `elem.lastChild` is `null`, so we can't access `elem.lastChild.nextSibling`. And the collection `elem.children` is empty (like an empty array `[]`). diff --git a/2-ui/1-document/03-dom-navigation/article.md b/2-ui/1-document/03-dom-navigation/article.md index 80054f9d7..b5f03098c 100644 --- a/2-ui/1-document/03-dom-navigation/article.md +++ b/2-ui/1-document/03-dom-navigation/article.md @@ -7,9 +7,9 @@ libs: # Walking the DOM -The DOM allows us to do anything with elements and their contents, but first we need to reach the corresponding DOM object, get it into a variable, and then we are able to modify it. +The DOM allows us to do anything with elements and their contents, but first we need to reach the corresponding DOM object. -All operations on the DOM start with the `document` object. From it we can access any node. +All operations on the DOM start with the `document` object. That's the main "entry point" to DOM. From it we can access any node. Here's a picture of links that allow for travel between DOM nodes: @@ -22,7 +22,7 @@ Let's discuss them in more detail. The topmost tree nodes are available directly as `document` properties: `<html>` = `document.documentElement` -: The topmost document node is `document.documentElement`. That's DOM node of `<html>` tag. +: The topmost document node is `document.documentElement`. That's the DOM node of the `<html>` tag. `<body>` = `document.body` : Another widely used DOM node is the `<body>` element -- `document.body`. @@ -86,9 +86,9 @@ For instance, here `<body>` has children `<div>` and `<ul>` (and few blank text </html> ``` -...And if we ask for all descendants of `<body>`, then we get direct children `<div>`, `<ul>` and also more nested elements like `<li>` (being a child of `<ul>`) and `<b>` (being a child of `<li>`) -- the entire subtree. +...And descendants of `<body>` are not only direct children `<div>`, `<ul>` but also more deeply nested elements, such as `<li>` (a child of `<ul>`) and `<b>` (a child of `<li>`) -- the entire subtree. -**The `childNodes` collection provides access to all child nodes, including text nodes.** +**The `childNodes` collection lists all child nodes, including text nodes.** The example below shows children of `document.body`: @@ -149,7 +149,7 @@ There are two important consequences: The first thing is nice. The second is tolerable, because we can use `Array.from` to create a "real" array from the collection, if we want array methods: ```js run - alert( Array.from(document.body.childNodes).filter ); // now it's there + alert( Array.from(document.body.childNodes).filter ); // function ``` ```warn header="DOM collections are read-only" @@ -182,35 +182,39 @@ Please, don't. The `for..in` loop iterates over all enumerable properties. And c ## Siblings and the parent -*Siblings* are nodes that are children of the same parent. For instance, `<head>` and `<body>` are siblings: +*Siblings* are nodes that are children of the same parent. + +For instance, here `<head>` and `<body>` are siblings: + +```html +<html> + <head>...</head><body>...</body> +</html> +``` - `<body>` is said to be the "next" or "right" sibling of `<head>`, - `<head>` is said to be the "previous" or "left" sibling of `<body>`. -The parent is available as `parentNode`. +The next sibling is in `nextSibling` property, and the previous one - in `previousSibling`. -The next node in the same parent (next sibling) is `nextSibling`, and the previous one is `previousSibling`. +The parent is available as `parentNode`. -For instance: +For example: -```html run -<html><head></head><body><script> - // HTML is "dense" to evade extra "blank" text nodes. - - // parent of <body> is <html> - alert( document.body.parentNode === document.documentElement ); // true +```js run +// parent of <body> is <html> +alert( document.body.parentNode === document.documentElement ); // true - // after <head> goes <body> - alert( document.head.nextSibling ); // HTMLBodyElement +// after <head> goes <body> +alert( document.head.nextSibling ); // HTMLBodyElement - // before <body> goes <head> - alert( document.body.previousSibling ); // HTMLHeadElement -</script></body></html> +// before <body> goes <head> +alert( document.body.previousSibling ); // HTMLHeadElement ``` ## Element-only navigation -Navigation properties listed above refer to *all* nodes. For instance, in `childNodes` we can see both text nodes, element nodes, and even comment nodes if there exist. +Navigation properties listed above refer to *all* nodes. For instance, in `childNodes` we can see both text nodes, element nodes, and even comment nodes if they exist. But for many tasks we don't want text or comment nodes. We want to manipulate element nodes that represent tags and form the structure of the page. @@ -222,7 +226,7 @@ The links are similar to those given above, just with `Element` word inside: - `children` -- only those children that are element nodes. - `firstElementChild`, `lastElementChild` -- first and last element children. -- `previousElementSibling`, `nextElementSibling` -- neighbour elements. +- `previousElementSibling`, `nextElementSibling` -- neighbor elements. - `parentElement` -- parent element. ````smart header="Why `parentElement`? Can the parent be *not* an element?" @@ -235,12 +239,12 @@ alert( document.documentElement.parentNode ); // document alert( document.documentElement.parentElement ); // null ``` -In other words, the `documentElement` (`<html>`) is the root node. Formally, it has `document` as its parent. But `document` is not an element node, so `parentNode` returns it and `parentElement` does not. +The reason is that the root node `document.documentElement` (`<html>`) has `document` as its parent. But `document` is not an element node, so `parentNode` returns it and `parentElement` does not. -This loop travels up from an arbitrary element `elem` to `<html>`, but not to the `document`: +This detail may be useful when we want to travel up from an arbitrary element `elem` to `<html>`, but not to the `document`: ```js -while(elem = elem.parentElement) { - alert( elem ); // parent chain till <html> +while(elem = elem.parentElement) { // go up till <html> + alert( elem ); } ``` ```` @@ -276,12 +280,12 @@ Till now we described the basic navigation properties. Certain types of DOM elements may provide additional properties, specific to their type, for convenience. -Tables are a great example and important particular case of that. +Tables are a great example of that, and represent a particularly important case: **The `<table>`** element supports (in addition to the given above) these properties: - `table.rows` -- the collection of `<tr>` elements of the table. - `table.caption/tHead/tFoot` -- references to elements `<caption>`, `<thead>`, `<tfoot>`. -- `table.tBodies` -- the collection of `<tbody>` elements (can be many according to the standard). +- `table.tBodies` -- the collection of `<tbody>` elements (can be many according to the standard, but there will always be at least one -- even if it is not in the source HTML, the browser will put it in the DOM). **`<thead>`, `<tfoot>`, `<tbody>`** elements provide the `rows` property: - `tbody.rows` -- the collection of `<tr>` inside. @@ -307,8 +311,9 @@ An example of usage: </table> <script> - // get the content of the first row, second cell - alert( table.*!*rows[0].cells[1]*/!*.innerHTML ) // "two" + // get td with "two" (first row, second column) + let td = table.*!*rows[0].cells[1]*/!*; + td.style.backgroundColor = "red"; // highlight it </script> ``` @@ -316,9 +321,9 @@ The specification: [tabular data](https://html.spec.whatwg.org/multipage/tables. There are also additional navigation properties for HTML forms. We'll look at them later when we start working with forms. -# Summary +## Summary -Given a DOM node, we can go to its immediate neighbours using navigation properties. +Given a DOM node, we can go to its immediate neighbors using navigation properties. There are two main sets of them: diff --git a/2-ui/1-document/04-searching-elements-dom/1-find-elements/solution.md b/2-ui/1-document/04-searching-elements-dom/1-find-elements/solution.md index c7080388e..c73aecd99 100644 --- a/2-ui/1-document/04-searching-elements-dom/1-find-elements/solution.md +++ b/2-ui/1-document/04-searching-elements-dom/1-find-elements/solution.md @@ -11,15 +11,15 @@ table.getElementsByTagName('label') // or document.querySelectorAll('#age-table label') -// 3. The first td in that table (with the word "Age"). +// 3. The first td in that table (with the word "Age") table.rows[0].cells[0] // or table.getElementsByTagName('td')[0] // or table.querySelector('td') -// 4. The form with the name "search". -// assuming there's only one element with name="search" +// 4. The form with the name "search" +// assuming there's only one element with name="search" in the document let form = document.getElementsByName('search')[0] // or, form specifically document.querySelector('form[name="search"]') @@ -29,8 +29,7 @@ form.getElementsByTagName('input')[0] // or form.querySelector('input') -// 6. The last input in that form. -// there's no direct query for that -let inputs = form.querySelectorAll('input') // search all -inputs[inputs.length-1] // take last +// 6. The last input in that form +let inputs = form.querySelectorAll('input') // find all inputs +inputs[inputs.length-1] // take the last one ``` diff --git a/2-ui/1-document/04-searching-elements-dom/1-find-elements/task.md b/2-ui/1-document/04-searching-elements-dom/1-find-elements/task.md index 4d4faeb2a..f0b54beac 100644 --- a/2-ui/1-document/04-searching-elements-dom/1-find-elements/task.md +++ b/2-ui/1-document/04-searching-elements-dom/1-find-elements/task.md @@ -6,12 +6,12 @@ importance: 4 Here's the document with the table and form. -How to find? +How to find?... 1. The table with `id="age-table"`. 2. All `label` elements inside that table (there should be 3 of them). 3. The first `td` in that table (with the word "Age"). -4. The `form` with the name `search`. +4. The `form` with `name="search"`. 5. The first `input` in that form. 6. The last `input` in that form. diff --git a/2-ui/1-document/04-searching-elements-dom/article.md b/2-ui/1-document/04-searching-elements-dom/article.md index 18ce24eba..405129694 100644 --- a/2-ui/1-document/04-searching-elements-dom/article.md +++ b/2-ui/1-document/04-searching-elements-dom/article.md @@ -6,9 +6,27 @@ There are additional searching methods for that. ## document.getElementById or just id -If an element has the `id` attribute, then there's a global variable by the name from that `id`. +If an element has the `id` attribute, we can get the element using the method `document.getElementById(id)`, no matter where it is. -We can use it to immediately access the element no matter where it is: +For instance: + +```html run +<div id="elem"> + <div id="elem-content">Element</div> +</div> + +<script> + // get the element +*!* + let elem = document.getElementById('elem'); +*/!* + + // make its background red + elem.style.background = 'red'; +</script> +``` + +Also, there's a global variable named by `id` that references the element: ```html run <div id="*!*elem*/!*"> @@ -16,57 +34,44 @@ We can use it to immediately access the element no matter where it is: </div> <script> - alert(elem); // DOM-element with id="elem" - alert(window.elem); // accessing global variable like this also works + // elem is a reference to DOM-element with id="elem" + elem.style.background = 'red'; - // for elem-content things are a bit more complex - // that has a dash inside, so it can't be a variable name - alert(window['elem-content']); // ...but accessible using square brackets [...] + // id="elem-content" has a hyphen inside, so it can't be a variable name + // ...but we can access it using square brackets: window['elem-content'] </script> ``` -The behavior is described [in the specification](http://www.whatwg.org/specs/web-apps/current-work/#dom-window-nameditem), but it is supported mainly for compatibility. The browser tries to help us by mixing namespaces of JS and DOM. Good for very simple scripts, but there may be name conflicts. Also, when we look in JS and don't have HTML in view, it's not obvious where the variable comes from. - -If we declare a variable with the same name, it takes precedence: +...That's unless we declare a JavaScript variable with the same name, then it takes precedence: ```html run untrusted height=0 <div id="elem"></div> <script> - let elem = 5; + let elem = 5; // now elem is 5, not a reference to <div id="elem"> - alert(elem); // the variable overrides the element + alert(elem); // 5 </script> ``` -The better alternative is to use a special method `document.getElementById(id)`. - -For instance: +```warn header="Please don't use id-named global variables to access elements" +This behavior is described [in the specification](https://html.spec.whatwg.org/multipage/window-object.html#named-access-on-the-window-object), but it is supported mainly for compatibility. -```html run -<div id="elem"> - <div id="elem-content">Element</div> -</div> +The browser tries to help us by mixing namespaces of JS and DOM. That's fine for simple scripts, inlined into HTML, but generally isn't a good thing. There may be naming conflicts. Also, when one reads JS code and doesn't have HTML in view, it's not obvious where the variable comes from. -<script> -*!* - let elem = document.getElementById('elem'); -*/!* +Here in the tutorial we use `id` to directly reference an element for brevity, when it's obvious where the element comes from. - elem.style.background = 'red'; -</script> +In real life `document.getElementById` is the preferred method. ``` -Here in the tutorial we'll often use `id` to directly reference an element, but that's only to keep things short. In real life `document.getElementById` is the preferred method. - -```smart header="There can be only one" +```smart header="The `id` must be unique" The `id` must be unique. There can be only one element in the document with the given `id`. -If there are multiple elements with the same `id`, then the behavior of corresponding methods is unpredictable. The browser may return any of them at random. So please stick to the rule and keep `id` unique. +If there are multiple elements with the same `id`, then the behavior of methods that use it is unpredictable, e.g. `document.getElementById` may return any of such elements at random. So please stick to the rule and keep `id` unique. ``` -```warn header="Only `document.getElementById`, not `anyNode.getElementById`" -The method `getElementById` that can be called only on `document` object. It looks for the given `id` in the whole document. +```warn header="Only `document.getElementById`, not `anyElem.getElementById`" +The method `getElementById` can be called only on `document` object. It looks for the given `id` in the whole document. ``` ## querySelectorAll [#querySelectorAll] @@ -98,22 +103,22 @@ Here we look for all `<li>` elements that are last children: This method is indeed powerful, because any CSS selector can be used. ```smart header="Can use pseudo-classes as well" -Pseudo-classes in the CSS selector like `:hover` and `:active` are also supported. For instance, `document.querySelectorAll(':hover')` will return the collection with elements that the pointer is over now (in nesting order: from the outermost `<html>` to the most nested one). +Pseudo-classes in the CSS selector like `:hover` and `:active` are also supported. For instance, `document.querySelectorAll(':hover')` will return the collection with elements that the pointer is over now (in nesting order: from the outermost `<html>` to the most nested one). ``` ## querySelector [#querySelector] The call to `elem.querySelector(css)` returns the first element for the given CSS selector. -In other words, the result is the same as `elem.querySelectorAll(css)[0]`, but the latter is looking for *all* elements and picking one, while `elem.querySelector` just looks for one. So it's faster and shorter to write. +In other words, the result is the same as `elem.querySelectorAll(css)[0]`, but the latter is looking for *all* elements and picking one, while `elem.querySelector` just looks for one. So it's faster and also shorter to write. ## matches Previous methods were searching the DOM. -The [elem.matches(css)](http://dom.spec.whatwg.org/#dom-element-matches) does not look for anything, it merely checks if `elem` matches the given CSS-selector. It returns `true` or `false`. +The [elem.matches(css)](https://dom.spec.whatwg.org/#dom-element-matches) does not look for anything, it merely checks if `elem` matches the given CSS-selector. It returns `true` or `false`. -The method comes handy when we are iterating over elements (like in array or something) and trying to filter those that interest us. +The method comes in handy when we are iterating over elements (like in an array or something) and trying to filter out those that interest us. For instance: @@ -137,7 +142,7 @@ For instance: *Ancestors* of an element are: parent, the parent of parent, its parent and so on. The ancestors together form the chain of parents from the element to the top. -The method `elem.closest(css)` looks the nearest ancestor that matches the CSS-selector. The `elem` itself is also included in the search. +The method `elem.closest(css)` looks for the nearest ancestor that matches the CSS-selector. The `elem` itself is also included in the search. In other words, the method `closest` goes up from the element and checks each of parents. If it matches the selector, then the search stops, and the ancestor is returned. @@ -149,7 +154,7 @@ For instance: <div class="contents"> <ul class="book"> <li class="chapter">Chapter 1</li> - <li class="chapter">Chapter 1</li> + <li class="chapter">Chapter 2</li> </ul> </div> @@ -172,8 +177,8 @@ Today, they are mostly history, as `querySelector` is more powerful and shorter So here we cover them mainly for completeness, while you can still find them in the old scripts. - `elem.getElementsByTagName(tag)` looks for elements with the given tag and returns the collection of them. The `tag` parameter can also be a star `"*"` for "any tags". -- `elem.getElementsByClassName(className)` returns elements that have the given CSS class. Elements may have other classes too. -- `document.getElementsByName(name)` returns elements with the given `name` attribute, document-wide. very rarely used. +- `elem.getElementsByClassName(className)` returns elements that have the given CSS class. +- `document.getElementsByName(name)` returns elements with the given `name` attribute, document-wide. Very rarely used. For instance: ```js @@ -305,8 +310,6 @@ If we use it instead, then both scripts output `1`: Now we can easily see the difference. The static collection did not increase after the appearance of a new `div` in the document. -Here we used separate scripts to illustrate how the element addition affects the collection, but any DOM manipulations affect them. Soon we'll see more of them. - ## Summary There are 6 main methods to search for nodes in DOM: @@ -360,12 +363,12 @@ There are 6 main methods to search for nodes in DOM: </tbody> </table> -By far the most used are `querySelector` and `querySelectorAll`, but `getElementBy*` can be sporadically helpful or found in the old scripts. +By far the most used are `querySelector` and `querySelectorAll`, but `getElement(s)By*` can be sporadically helpful or found in the old scripts. Besides that: - There is `elem.matches(css)` to check if `elem` matches the given CSS selector. - There is `elem.closest(css)` to look for the nearest ancestor that matches the given CSS-selector. The `elem` itself is also checked. -And let's mention one more method here to check for the child-parent relationship: +And let's mention one more method here to check for the child-parent relationship, as it's sometimes useful: - `elemA.contains(elemB)` returns true if `elemB` is inside `elemA` (a descendant of `elemA`) or when `elemA==elemB`. diff --git a/2-ui/1-document/05-basic-dom-node-properties/2-tree-info/solution.md b/2-ui/1-document/05-basic-dom-node-properties/2-tree-info/solution.md index 781b7a929..0088882c2 100644 --- a/2-ui/1-document/05-basic-dom-node-properties/2-tree-info/solution.md +++ b/2-ui/1-document/05-basic-dom-node-properties/2-tree-info/solution.md @@ -6,7 +6,9 @@ for (let li of document.querySelectorAll('li')) { } ``` -In the loop we need to get the text inside every `li`. We can read it directly from the first child node, that is the text node: +In the loop we need to get the text inside every `li`. + +We can read the text from the first child node of `li`, that is the text node: ```js for (let li of document.querySelectorAll('li')) { @@ -16,4 +18,4 @@ for (let li of document.querySelectorAll('li')) { } ``` -Then we can get the number of descendants `li.getElementsByTagName('li')`. +Then we can get the number of descendants as `li.getElementsByTagName('li').length`. diff --git a/2-ui/1-document/05-basic-dom-node-properties/4-where-document-in-hierarchy/solution.md b/2-ui/1-document/05-basic-dom-node-properties/4-where-document-in-hierarchy/solution.md index db7ebc9d8..cb9456717 100644 --- a/2-ui/1-document/05-basic-dom-node-properties/4-where-document-in-hierarchy/solution.md +++ b/2-ui/1-document/05-basic-dom-node-properties/4-where-document-in-hierarchy/solution.md @@ -27,7 +27,7 @@ Also, there's a reference to the constructor function inside the `prototype`: alert(HTMLDocument.prototype.constructor === HTMLDocument); // true ``` -For built-in classes in all prototypes there's a `constructor` reference, and we can get `constructor.name` to see the name of the class. Let's do it for all objects in the `document` prototype chain: +To get a name of the class as a string, we can use `constructor.name`. Let's do it for the whole `document` prototype chain, till class `Node`: ```js run alert(HTMLDocument.prototype.constructor.name); // HTMLDocument @@ -35,4 +35,6 @@ alert(HTMLDocument.prototype.__proto__.constructor.name); // Document alert(HTMLDocument.prototype.__proto__.__proto__.constructor.name); // Node ``` +That's the hierarchy. + We also could examine the object using `console.dir(document)` and see these names by opening `__proto__`. The console takes them from `constructor` internally. diff --git a/2-ui/1-document/05-basic-dom-node-properties/article.md b/2-ui/1-document/05-basic-dom-node-properties/article.md index e43040ba7..99dde5bcd 100644 --- a/2-ui/1-document/05-basic-dom-node-properties/article.md +++ b/2-ui/1-document/05-basic-dom-node-properties/article.md @@ -2,15 +2,15 @@ Let's get a more in-depth look at DOM nodes. -In this chapter we'll see more into what they are and their most used properties. +In this chapter we'll see more into what they are and learn their most used properties. ## DOM node classes -DOM nodes have different properties depending on their class. For instance, an element node corresponding to tag `<a>` has link-related properties, and the one corresponding to `<input>` has input-related properties and so on. Text nodes are not the same as element nodes. But there are also common properties and methods between all of them, because all classes of DOM nodes form a single hierarchy. +Different DOM nodes may have different properties. For instance, an element node corresponding to tag `<a>` has link-related properties, and the one corresponding to `<input>` has input-related properties and so on. Text nodes are not the same as element nodes. But there are also common properties and methods between all of them, because all classes of DOM nodes form a single hierarchy. Each DOM node belongs to the corresponding built-in class. -The root of the hierarchy is [EventTarget](https://dom.spec.whatwg.org/#eventtarget), that is inherited by [Node](http://dom.spec.whatwg.org/#interface-node), and other DOM nodes inherit from it. +The root of the hierarchy is [EventTarget](https://dom.spec.whatwg.org/#eventtarget), that is inherited by [Node](https://dom.spec.whatwg.org/#interface-node), and other DOM nodes inherit from it. Here's the picture, explanations to follow: @@ -18,27 +18,52 @@ Here's the picture, explanations to follow: The classes are: -- [EventTarget](https://dom.spec.whatwg.org/#eventtarget) -- is the root "abstract" class. Objects of that class are never created. It serves as a base, so that all DOM nodes support so-called "events", we'll study them later. -- [Node](http://dom.spec.whatwg.org/#interface-node) -- is also an "abstract" class, serving as a base for DOM nodes. It provides the core tree functionality: `parentNode`, `nextSibling`, `childNodes` and so on (they are getters). Objects of `Node` class are never created. But there are concrete node classes that inherit from it, namely: `Text` for text nodes, `Element` for element nodes and more exotic ones like `Comment` for comment nodes. -- [Element](http://dom.spec.whatwg.org/#interface-element) -- is a base class for DOM elements. It provides element-level navigation like `nextElementSibling`, `children` and searching methods like `getElementsByTagName`, `querySelector`. In the browser there may be not only HTML, but also XML and SVG documents. The `Element` class serves as a base for more specific classes: `SVGElement`, `XMLElement` and `HTMLElement`. -- [HTMLElement](https://html.spec.whatwg.org/multipage/dom.html#htmlelement) -- is finally the basic class for all HTML elements. It is inherited by various HTML elements: +- [EventTarget](https://dom.spec.whatwg.org/#eventtarget) -- is the root "abstract" class for everything. + + Objects of that class are never created. It serves as a base, so that all DOM nodes support so-called "events", we'll study them later. + +- [Node](https://dom.spec.whatwg.org/#interface-node) -- is also an "abstract" class, serving as a base for DOM nodes. + + It provides the core tree functionality: `parentNode`, `nextSibling`, `childNodes` and so on (they are getters). Objects of `Node` class are never created. But there are other classes that inherit from it (and so inherit the `Node` functionality). + +- [Document](https://dom.spec.whatwg.org/#interface-document), for historical reasons often inherited by `HTMLDocument` (though the latest spec doesn't dictate it) -- is a document as a whole. + + The `document` global object belongs exactly to this class. It serves as an entry point to the DOM. + +- [CharacterData](https://dom.spec.whatwg.org/#interface-characterdata) -- an "abstract" class, inherited by: + - [Text](https://dom.spec.whatwg.org/#interface-text) -- the class corresponding to a text inside elements, e.g. `Hello` in `<p>Hello</p>`. + - [Comment](https://dom.spec.whatwg.org/#interface-comment) -- the class for comments. They are not shown, but each comment becomes a member of DOM. + +- [Element](https://dom.spec.whatwg.org/#interface-element) -- is the base class for DOM elements. + + It provides element-level navigation like `nextElementSibling`, `children` and searching methods like `getElementsByTagName`, `querySelector`. + + A browser supports not only HTML, but also XML and SVG. So the `Element` class serves as a base for more specific classes: `SVGElement`, `XMLElement` (we don't need them here) and `HTMLElement`. + +- Finally, [HTMLElement](https://html.spec.whatwg.org/multipage/dom.html#htmlelement) is the basic class for all HTML elements. We'll work with it most of the time. + + It is inherited by concrete HTML elements: - [HTMLInputElement](https://html.spec.whatwg.org/multipage/forms.html#htmlinputelement) -- the class for `<input>` elements, - [HTMLBodyElement](https://html.spec.whatwg.org/multipage/semantics.html#htmlbodyelement) -- the class for `<body>` elements, - - [HTMLAnchorElement](https://html.spec.whatwg.org/multipage/semantics.html#htmlanchorelement) -- the class for `<a>` elements - - ...and so on, each tag has its own class that may provide specific properties and methods. + - [HTMLAnchorElement](https://html.spec.whatwg.org/multipage/semantics.html#htmlanchorelement) -- the class for `<a>` elements, + - ...and so on. + +There are many other tags with their own classes that may have specific properties and methods, while some elements, such as `<span>`, `<section>`, `<article>` do not have any specific properties, so they are instances of `HTMLElement` class. + +So, the full set of properties and methods of a given node comes as the result of the chain of inheritance. -So, the full set of properties and methods of a given node comes as the result of the inheritance. +For example, let's consider the DOM object for an `<input>` element. It belongs to [HTMLInputElement](https://html.spec.whatwg.org/multipage/forms.html#htmlinputelement) class. -For example, let's consider the DOM object for an `<input>` element. It belongs to [HTMLInputElement](https://html.spec.whatwg.org/multipage/forms.html#htmlinputelement) class. It gets properties and methods as a superposition of: +It gets properties and methods as a superposition of (listed in inheritance order): -- `HTMLInputElement` -- this class provides input-specific properties, and inherits from... -- `HTMLElement` -- it provides common HTML element methods (and getters/setters) and inherits from... -- `Element` -- provides generic element methods and inherits from... -- `Node` -- provides common DOM node properties and inherits from... +- `HTMLInputElement` -- this class provides input-specific properties, +- `HTMLElement` -- it provides common HTML element methods (and getters/setters), +- `Element` -- provides generic element methods, +- `Node` -- provides common DOM node properties, - `EventTarget` -- gives the support for events (to be covered), -- ...and finally it inherits from `Object`, so "pure object" methods like `hasOwnProperty` are also available. +- ...and finally it inherits from `Object`, so "plain object" methods like `hasOwnProperty` are also available. -To see the DOM node class name, we can recall that an object usually has the `constructor` property. It references to the class constructor, and `constructor.name` is its name: +To see the DOM node class name, we can recall that an object usually has the `constructor` property. It references the class constructor, and `constructor.name` is its name: ```js run alert( document.body.constructor.name ); // HTMLBodyElement @@ -76,7 +101,7 @@ Try it on `document.body`. ``` ````smart header="IDL in the spec" -In the specification, classes are described not using JavaScript, but a special [Interface description language](https://en.wikipedia.org/wiki/Interface_description_language) (IDL), that is usually easy to understand. +In the specification, DOM classes aren't described by using JavaScript, but a special [Interface description language](https://en.wikipedia.org/wiki/Interface_description_language) (IDL), that is usually easy to understand. In IDL all properties are prepended with their types. For instance, `DOMString`, `boolean` and so on. @@ -91,7 +116,7 @@ interface HTMLInputElement: HTMLElement { // here go properties and methods of <input> elements *!* - // "DOMString" means that the value of these properties are strings + // "DOMString" means that the value of a property is a string */!* attribute DOMString accept; attribute DOMString alt; @@ -110,13 +135,11 @@ interface HTMLInputElement: HTMLElement { ... } ``` - -Other classes are somewhat similar. ```` ## The "nodeType" property -The `nodeType` property provides an old-fashioned way to get the "type" of a DOM node. +The `nodeType` property provides one more, "old-fashioned" way to get the "type" of a DOM node. It has a numeric value: - `elem.nodeType == 1` for element nodes, @@ -128,13 +151,13 @@ For instance: ```html run <body> - <script> + <script> let elem = document.body; - // let's examine what it is? + // let's examine: what type of node is in elem? alert(elem.nodeType); // 1 => element - // and the first child is... + // and its first child is... alert(elem.firstChild.nodeType); // 3 => text // for the document object, the type is 9 @@ -156,7 +179,7 @@ alert( document.body.nodeName ); // BODY alert( document.body.tagName ); // BODY ``` -Is there any difference between tagName and nodeName? +Is there any difference between `tagName` and `nodeName`? Sure, the difference is reflected in their names, but is indeed a bit subtle. @@ -175,20 +198,19 @@ For instance, let's compare `tagName` and `nodeName` for the `document` and a co <script> // for comment - alert( document.body.firstChild.tagName ); // undefined (no element) + alert( document.body.firstChild.tagName ); // undefined (not an element) alert( document.body.firstChild.nodeName ); // #comment // for document - alert( document.tagName ); // undefined (not element) + alert( document.tagName ); // undefined (not an element) alert( document.nodeName ); // #document </script> </body> ``` -If we only deal with elements, then `tagName` is the only thing we should use. +If we only deal with elements, then we can use both `tagName` and `nodeName` - there's no difference. - -```smart header="The tag name is always uppercase except XHTML" +```smart header="The tag name is always uppercase except in XML mode" The browser has two modes of processing documents: HTML and XML. Usually the HTML-mode is used for webpages. XML-mode is enabled when the browser receives an XML-document with the header: `Content-Type: application/xml+xhtml`. In HTML mode `tagName/nodeName` is always uppercased: it's `BODY` either for `<body>` or `<BoDy>`. @@ -199,9 +221,9 @@ In XML mode the case is kept "as is". Nowadays XML mode is rarely used. ## innerHTML: the contents -The [innerHTML](https://w3c.github.io/DOM-Parsing/#widl-Element-innerHTML) property allows to get the HTML inside the element as a string. +The [innerHTML](https://w3c.github.io/DOM-Parsing/#the-innerhtml-mixin) property allows to get the HTML inside the element as a string. -We can also modify it. So it's one of most powerful ways to change the page. +We can also modify it. So it's one of the most powerful ways to change the page. The example shows the contents of `document.body` and then replaces it completely: @@ -232,14 +254,12 @@ We can try to insert invalid HTML, the browser will fix our errors: ``` ```smart header="Scripts don't execute" -If `innerHTML` inserts a `<script>` tag into the document -- it doesn't execute. - -It becomes a part of HTML, just as a script that has already run. +If `innerHTML` inserts a `<script>` tag into the document -- it becomes a part of HTML, but doesn't execute. ``` ### Beware: "innerHTML+=" does a full overwrite -We can append "more HTML" by using `elem.innerHTML+="something"`. +We can append HTML to an element by using `elem.innerHTML+="more html"`. Like this: @@ -287,7 +307,7 @@ Here's an example: </script> ``` -**Beware: unlike `innerHTML`, writing to `outerHTML` does not change the element. Instead, it replaces it as a whole in the outer context.** +**Beware: unlike `innerHTML`, writing to `outerHTML` does not change the element. Instead, it replaces it in the DOM.** Yeah, sounds strange, and strange it is, that's why we make a separate note about it here. Take a look. @@ -302,32 +322,37 @@ Consider the example: *!* // replace div.outerHTML with <p>...</p> */!* - div.outerHTML = '<p>A new element!</p>'; // (*) + div.outerHTML = '<p>A new element</p>'; // (*) *!* - // Wow! The div is still the same! + // Wow! 'div' is still the same! */!* - alert(div.outerHTML); // <div>Hello, world!</div> + alert(div.outerHTML); // <div>Hello, world!</div> (**) </script> ``` -In the line `(*)` we take the full HTML of `<div>...</div>` and replace it by `<p>...</p>`. In the outer document we can see the new content instead of the `<div>`. But the old `div` variable is still the same. +Looks really odd, right? -The `outerHTML` assignment does not modify the DOM element, but extracts it from the outer context and inserts a new piece of HTML instead of it. +In the line `(*)` we replaced `div` with `<p>A new element</p>`. In the outer document (the DOM) we can see the new content instead of the `<div>`. But, as we can see in line `(**)`, the value of the old `div` variable hasn't changed! -Novice developers sometimes make an error here: they modify `div.outerHTML` and then continue to work with `div` as if it had the new content in it. +The `outerHTML` assignment does not modify the DOM element (the object referenced by, in this case, the variable 'div'), but removes it from the DOM and inserts the new HTML in its place. -That's possible with `innerHTML`, but not with `outerHTML`. +So what happened in `div.outerHTML=...` is: +- `div` was removed from the document. +- Another piece of HTML `<p>A new element</p>` was inserted in its place. +- `div` still has its old value. The new HTML wasn't saved to any variable. -We can write to `outerHTML`, but should keep in mind that it doesn't change the element we're writing to. It creates the new content on its place instead. We can get a reference to new elements by querying DOM. +It's so easy to make an error here: modify `div.outerHTML` and then continue to work with `div` as if it had the new content in it. But it doesn't. Such thing is correct for `innerHTML`, but not for `outerHTML`. + +We can write to `elem.outerHTML`, but should keep in mind that it doesn't change the element we're writing to ('elem'). It puts the new HTML in its place instead. We can get references to the new elements by querying the DOM. ## nodeValue/data: text node content The `innerHTML` property is only valid for element nodes. -Other node types have their counterpart: `nodeValue` and `data` properties. These two are almost the same for practical use, there are only minor specification differences. So we'll use `data`, because it's shorter. +Other node types, such as text nodes, have their counterpart: `nodeValue` and `data` properties. These two are almost the same for practical use, there are only minor specification differences. So we'll use `data`, because it's shorter. -We can read it, like this: +An example of reading the content of a text node and a comment: ```html run height="50" <body> @@ -347,7 +372,9 @@ We can read it, like this: </body> ``` -For text nodes we can imagine a reason to read or modify them, but why comments? Usually, they are not interesting at all, but sometimes developers embed information into HTML in them, like this: +For text nodes we can imagine a reason to read or modify them, but why comments? + +Sometimes developers embed information or template instructions into HTML in them, like this: ```html <!-- if isAdmin --> @@ -355,7 +382,7 @@ For text nodes we can imagine a reason to read or modify them, but why comments? <!-- /if --> ``` -...Then JavaScript can read it and process embedded instructions. +...Then JavaScript can read it from `data` property and process embedded instructions. ## textContent: pure text @@ -393,7 +420,7 @@ Compare the two: <div id="elem2"></div> <script> - let name = prompt("What's your name?", "<b>Winnie-the-pooh!</b>"); + let name = prompt("What's your name?", "<b>Winnie-the-Pooh!</b>"); elem1.innerHTML = name; elem2.textContent = name; @@ -401,7 +428,7 @@ Compare the two: ``` 1. The first `<div>` gets the name "as HTML": all tags become tags, so we see the bold name. -2. The second `<div>` gets the name "as text", so we literally see `<b>Winnie-the-pooh!</b>`. +2. The second `<div>` gets the name "as text", so we literally see `<b>Winnie-the-Pooh!</b>`. In most cases, we expect the text from a user, and want to treat it as text. We don't want unexpected HTML in our site. An assignment to `textContent` does exactly that. @@ -409,7 +436,7 @@ In most cases, we expect the text from a user, and want to treat it as text. We The "hidden" attribute and the DOM property specifies whether the element is visible or not. -We can use it in HTML or assign using JavaScript, like this: +We can use it in HTML or assign it using JavaScript, like this: ```html run height="80" <div>Both divs below are hidden</div> @@ -438,7 +465,7 @@ Here's a blinking element: ## More properties -DOM elements also have additional properties, many of them provided by the class: +DOM elements also have additional properties, in particular those that depend on the class: - `value` -- the value for `<input>`, `<select>` and `<textarea>` (`HTMLInputElement`, `HTMLSelectElement`...). - `href` -- the "href" for `<a href="...">` (`HTMLAnchorElement`). @@ -459,7 +486,7 @@ For instance: Most standard HTML attributes have the corresponding DOM property, and we can access it like that. -If we want to know the full list of supported properties for a given class, we can find them in the specification. For instance, HTMLInputElement is documented at <https://html.spec.whatwg.org/#htmlinputelement>. +If we want to know the full list of supported properties for a given class, we can find them in the specification. For instance, `HTMLInputElement` is documented at <https://html.spec.whatwg.org/#htmlinputelement>. Or if we'd like to get them fast or are interested in a concrete browser specification -- we can always output the element using `console.dir(elem)` and read the properties. Or explore "DOM properties" in the Elements tab of the browser developer tools. @@ -470,7 +497,7 @@ Each DOM node belongs to a certain class. The classes form a hierarchy. The full Main DOM node properties are: `nodeType` -: We can get `nodeType` from the DOM object class, but often we need just to see if it is a text or element node. The `nodeType` property is good for that. It has numeric values, most important are: `1` -- for elements,`3` -- for text nodes. Read-only. +: We can use it to see if a node is a text or an element node. It has a numeric value: `1` for elements,`3` for text nodes, and a few others for other node types. Read-only. `nodeName/tagName` : For elements, tag name (uppercased unless XML-mode). For non-element nodes `nodeName` describes what it is. Read-only. @@ -485,11 +512,11 @@ Main DOM node properties are: : The content of a non-element node (text, comment). These two are almost the same, usually we use `data`. Can be modified. `textContent` -: The text inside the element, basically HTML minus all `<tags>`. Writing into it puts the text inside the element, with all special characters and tags treated exactly as text. Can safely insert user-generated text and protect from unwanted HTML insertions. +: The text inside the element: HTML minus all `<tags>`. Writing into it puts the text inside the element, with all special characters and tags treated exactly as text. Can safely insert user-generated text and protect from unwanted HTML insertions. `hidden` : When set to `true`, does the same as CSS `display:none`. DOM nodes also have other properties depending on their class. For instance, `<input>` elements (`HTMLInputElement`) support `value`, `type`, while `<a>` elements (`HTMLAnchorElement`) support `href` etc. Most standard HTML attributes have a corresponding DOM property. -But HTML attributes and DOM properties are not always the same, as we'll see in the next chapter. +However, HTML attributes and DOM properties are not always the same, as we'll see in the next chapter. diff --git a/2-ui/1-document/05-basic-dom-node-properties/dom-class-hierarchy.svg b/2-ui/1-document/05-basic-dom-node-properties/dom-class-hierarchy.svg index 39f7d8f8c..4a6705272 100644 --- a/2-ui/1-document/05-basic-dom-node-properties/dom-class-hierarchy.svg +++ b/2-ui/1-document/05-basic-dom-node-properties/dom-class-hierarchy.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="478" height="364" viewBox="0 0 478 364"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="dom-class-hierarchy.svg"><path id="Rectangle-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 6h118v28H181z"/><path id="Rectangle-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 74h118v28H181z"/><text id="EventTarget" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.9" y="24">EventTarget</tspan></text><text id="Node" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="225.6" y="91">Node </tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M240.5 39.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 144h118v28H181z"/><text id="Element" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="214.8" y="161">Element </tspan></text><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M240.5 109.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 230h118v28H181z"/><text id="HTMLElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.4" y="247">HTMLElement </tspan></text><path id="Line-2-Copy-4" fill="#C06334" fill-rule="nonzero" d="M240.5 195.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M171 300h138v28H171z"/><text id="HTMLBodyElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="186" y="317">HTMLBodyElement </tspan></text><path id="Line-2-Copy-6" fill="#C06334" fill-rule="nonzero" d="M240.5 265.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M12 300h138v28H12z"/><text id="HTMLInputElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="24.4" y="317">HTMLInputElement </tspan></text><path id="Line-2-Copy-7" fill="#C06334" fill-rule="nonzero" d="M170 261l-6.753 14.12-3.685-4.736-29.448 22.905-.79.614-1.227-1.578.79-.614 29.448-22.906-3.684-4.735L170 261z"/><path id="Rectangle-8-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M332 300h138v28H332z"/><text id="HTMLAnchorElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="341.3" y="317">HTMLAnchorElement </tspan></text><path id="Line-2-Copy-8" fill="#C06334" fill-rule="nonzero" d="M307 259l14.554 5.76-4.47 4.002 20.661 23.07.667.746-1.49 1.334-.667-.745-20.66-23.071-4.47 4.003L307 259z"/><path id="Rectangle-8-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M45 126h118v28H45z"/><text id="Text" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="89.6" y="144">Text </tspan></text><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M171 96l-6.589 14.198-3.738-4.693-18.55 14.777-.782.623-1.246-1.564.782-.623 18.549-14.778-3.738-4.692L171 96z"/><path id="Rectangle-8-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M315 126h118v28H315z"/><text id="Comment" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.8" y="144">Comment </tspan></text><path id="Line-2-Copy-3" fill="#C06334" fill-rule="nonzero" d="M307 96l15.338 3.123-3.701 4.723 18.98 14.867.787.616-1.233 1.575-.788-.617-18.979-14.867-3.7 4.724L307 96z"/><path id="Line-2-Copy-9" fill="#C06334" fill-rule="nonzero" d="M307 174l15.338 3.123-3.701 4.723 18.98 14.867.787.616-1.233 1.575-.788-.617-18.979-14.867-3.7 4.724L307 174z"/><path id="Rectangle-8-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M315 205h118v28H315z"/><text id="SVGElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="338" y="223">SVGElement </tspan></text><text id="<div>Text</div>" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="48" y="171" fill="#AF6E24"><div></tspan> <tspan x="84" y="171" fill="#C06334">Text</tspan> <tspan x="112.8" y="171" fill="#AF6E24"></div></tspan></text><text id="<input-type="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="24.4" y="341"><input type="…"></tspan></text><text id="<body>" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="220.4" y="341"><body></tspan></text><text id="<a-href="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="358.8" y="341"><a href="…"></tspan></text><text id="<div>Text</div>" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="189" y="188" fill="#AF6E24"><</tspan> <tspan x="196.2" y="188" fill="#C06334">div</tspan> <tspan x="217.8" y="188" fill="#AF6E24">></tspan> <tspan x="225" y="188" fill="#DBAF88">Text</tspan> <tspan x="253.8" y="188" fill="#AF6E24"></</tspan> <tspan x="268.2" y="188" fill="#C06334">div</tspan> <tspan x="289.8" y="188" fill="#AF6E24">></tspan></text><text id="<!--comment-->" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="327.6" y="171"><!--comment--></tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="478" height="364" viewBox="0 0 478 364"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="dom-class-hierarchy.svg"><path id="Rectangle-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 6h118v28H181z"/><path id="Rectangle-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 74h118v28H181z"/><text id="EventTarget" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.9" y="24">EventTarget</tspan></text><text id="Node" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="225.6" y="91">Node </tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M240.5 39.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 144h118v28H181z"/><text id="Element" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="214.8" y="161">Element </tspan></text><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M240.5 109.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 230h118v28H181z"/><text id="HTMLElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.4" y="247">HTMLElement </tspan></text><path id="Line-2-Copy-4" fill="#C06334" fill-rule="nonzero" d="M240.5 195.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M171 300h138v28H171z"/><text id="HTMLBodyElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="186" y="317">HTMLBodyElement </tspan></text><path id="Line-2-Copy-6" fill="#C06334" fill-rule="nonzero" d="M240.5 265.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M12 300h138v28H12z"/><text id="HTMLInputElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="24.4" y="317">HTMLInputElement </tspan></text><path id="Line-2-Copy-7" fill="#C06334" fill-rule="nonzero" d="M170 261l-6.753 14.12-3.685-4.736-29.448 22.905-.79.614-1.227-1.578.79-.614 29.448-22.906-3.684-4.735L170 261z"/><path id="Rectangle-8-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M332 300h138v28H332z"/><text id="HTMLAnchorElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="341.3" y="317">HTMLAnchorElement </tspan></text><path id="Line-2-Copy-8" fill="#C06334" fill-rule="nonzero" d="M307 259l14.554 5.76-4.47 4.002 20.661 23.07.667.746-1.49 1.334-.667-.745-20.66-23.071-4.47 4.003L307 259z"/><path id="Rectangle-8-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M45 126h118v28H45z"/><text id="Text" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="89.6" y="144">Text </tspan></text><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M171 96l-6.589 14.198-3.738-4.693-18.55 14.777-.782.623-1.246-1.564.782-.623 18.549-14.778-3.738-4.692L171 96z"/><path id="Rectangle-8-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M315 126h118v28H315z"/><text id="Comment" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.8" y="144">Comment </tspan></text><path id="Line-2-Copy-3" fill="#C06334" fill-rule="nonzero" d="M307 96l15.338 3.123-3.701 4.723 18.98 14.867.787.616-1.233 1.575-.788-.617-18.979-14.867-3.7 4.724L307 96z"/><path id="Line-2-Copy-9" fill="#C06334" fill-rule="nonzero" d="M307 174l15.338 3.123-3.701 4.723 18.98 14.867.787.616-1.233 1.575-.788-.617-18.979-14.867-3.7 4.724L307 174z"/><path id="Rectangle-8-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M315 205h118v28H315z"/><text id="SVGElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="338" y="223">SVGElement </tspan></text><text id="<div>Text</div>" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="48" y="171" fill="#AF6E24"><div></tspan> <tspan x="84" y="171" fill="#C06334">Text</tspan> <tspan x="112.8" y="171" fill="#AF6E24"></div></tspan></text><text id="<input-type="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="24.4" y="341"><input type="…"></tspan></text><text id="<body>" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="220.4" y="341"><body></tspan></text><text id="<a-href="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="358.8" y="341"><a href="…"></tspan></text><text id="<div>Text</div>" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="189" y="188" fill="#AF6E24"><</tspan> <tspan x="196.2" y="188" fill="#C06334">div</tspan> <tspan x="217.8" y="188" fill="#AF6E24">></tspan> <tspan x="225" y="188" fill="#DBAF88">Text</tspan> <tspan x="253.8" y="188" fill="#AF6E24"></</tspan> <tspan x="268.2" y="188" fill="#C06334">div</tspan> <tspan x="289.8" y="188" fill="#AF6E24">></tspan></text><text id="<!--comment-->" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="327.6" y="171"><!--comment--></tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="552" height="403" viewBox="0 0 552 403"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="dom-class-hierarchy.svg"><path id="Rectangle-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 6h118v28H181z"/><path id="Rectangle-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 74h118v28H181z"/><text id="EventTarget" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.9" y="24">EventTarget</tspan></text><text id="Node" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="225.6" y="91">Node </tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M240.5 39.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 144h118v28H181z"/><text id="Element" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="214.8" y="161">Element </tspan></text><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M240.5 109.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-4" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M181 230h118v28H181z"/><text id="HTMLElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="200.4" y="247">HTMLElement </tspan></text><path id="Line-2-Copy-4" fill="#C06334" fill-rule="nonzero" d="M240.5 195.5l7 14h-6v17h-2v-17h-6l7-14zM72.5 158.5l7.273 13.86-5.999.117L74 183.98l.02 1-2 .04-.02-1-.226-11.503-5.998.118L72.5 158.5z"/><path id="Rectangle-8-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M171 300h138v28H171z"/><text id="HTMLBodyElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="186" y="317">HTMLBodyElement </tspan></text><path id="Line-2-Copy-6" fill="#C06334" fill-rule="nonzero" d="M240.5 265.5l7 14h-6v17h-2v-17h-6l7-14z"/><path id="Rectangle-8-Copy-7" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 300h138v28H1z"/><text id="HTMLInputElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="13.4" y="317">HTMLInputElement </tspan></text><path id="Line-2-Copy-7" fill="#C06334" fill-rule="nonzero" d="M159 261l-6.753 14.12-3.685-4.736-29.448 22.905-.79.614-1.227-1.578.79-.614 29.448-22.906-3.684-4.735L159 261z"/><path id="Rectangle-8-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M339 300h138v28H339z"/><text id="HTMLAnchorElement" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.3" y="317">HTMLAnchorElement </tspan></text><path id="Line-2-Copy-8" fill="#C06334" fill-rule="nonzero" d="M312 261l15.305 3.28-3.749 4.684 29.069 23.255.78.625-1.249 1.562-.78-.625-29.069-23.254-3.748 4.685L312 261z"/><path id="Rectangle-8-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M22 126h98v28H22z"/><text id="Document" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="42.2" y="144">Document </tspan></text><path id="Rectangle-8-Copy-2" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M22 192h98v28H22z"/><text id="HTMLDocument" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="27.8" y="210">HTMLDocument </tspan></text><path id="Line-2-Copy-2" fill="#C06334" fill-rule="nonzero" d="M168 90l-8.862 12.902-2.905-5.251-34.749 19.224-.875.484-.968-1.75.875-.484 34.749-19.224-2.904-5.25L168 90z"/><path id="Rectangle-8-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M369 126h118v28H369z"/><text id="CharacterData" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="381.2" y="144">CharacterData </tspan></text><path id="Line-2-Copy-3" fill="#C06334" fill-rule="nonzero" d="M314 91l15.648.379-2.813 5.299 36.634 19.439.883.468-.937 1.767-.884-.469-36.633-19.439-2.812 5.301L314 91zM458 159l13.93 7.138-4.836 3.552 9.712 13.218.592.806-1.612 1.184-.592-.806-9.712-13.218-4.834 3.553L458 159zM400 158l-1.376 15.592-5.11-3.146-8.662 14.078-.524.852-1.704-1.048.524-.852 8.663-14.078-5.11-3.143L400 158z"/><text id="Document-as-a-whole" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="6.6" y="235">Document as a whole</tspan></text><text id="<input-type="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="13.4" y="341"><input type="…"></tspan></text><text id="<body>" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="220.4" y="341"><body></tspan></text><text id="<a-href="…">" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="365.8" y="341"><a href="…"></tspan></text><text id="<div>...</div>" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="192.6" y="188" fill="#AF6E24"><</tspan> <tspan x="199.8" y="188" fill="#C06334">div</tspan> <tspan x="221.4" y="188" fill="#AF6E24">></tspan> <tspan x="228.6" y="188" fill="#DBAF88">...</tspan> <tspan x="250.2" y="188" fill="#AF6E24"></</tspan> <tspan x="264.6" y="188" fill="#C06334">div</tspan> <tspan x="286.2" y="188" fill="#AF6E24">></tspan></text><path id="Rectangle-8-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M449 192h78v28h-78z"/><text id="Comment" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="463.8" y="210">Comment </tspan></text><text id="<!--comment-->" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="442.6" y="237"><!--comment--></tspan></text><path id="Rectangle-8-Copy-3" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M342 192h78v28h-78z"/><text id="Text" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="365.6" y="210">Text </tspan></text><text id=""Hello"" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="356.8" y="237">"Hello"</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/06-dom-attributes-and-properties/article.md b/2-ui/1-document/06-dom-attributes-and-properties/article.md index c1bcf2ce1..b02f626dc 100644 --- a/2-ui/1-document/06-dom-attributes-and-properties/article.md +++ b/2-ui/1-document/06-dom-attributes-and-properties/article.md @@ -8,7 +8,7 @@ But the attribute-property mapping is not one-to-one! In this chapter we'll pay ## DOM properties -We've already seen built-in DOM properties. There's a lot. But technically no one limits us, and if it's not enough -- we can add our own. +We've already seen built-in DOM properties. There are a lot. But technically no one limits us, and if there aren't enough, we can add our own. DOM nodes are regular JavaScript objects. We can alter them. @@ -124,7 +124,7 @@ Here's an extended demo of working with attributes: elem.setAttribute('Test', 123); // (2), writing - alert( elem.outerHTML ); // (3), see it's there + alert( elem.outerHTML ); // (3), see if the attribute is in HTML (yes) for (let attr of elem.attributes) { // (4) list all alert( `${attr.name} = ${attr.value}` ); @@ -162,7 +162,7 @@ In the example below `id` is modified as an attribute, and we can see the proper </script> ``` -But there are exclusions, for instance `input.value` synchronizes only from attribute -> to property, but not back: +But there are exclusions, for instance `input.value` synchronizes only from attribute -> property, but not back: ```html run <input> @@ -186,7 +186,7 @@ In the example above: - Changing the attribute `value` updates the property. - But the property change does not affect the attribute. -That "feature" may actually come in handy, because the user may modify `value`, and then after it, if we want to recover the "original" value from HTML, it's in the attribute. +That "feature" may actually come in handy, because the user actions may lead to `value` changes, and then after them, if we want to recover the "original" value from HTML, it's in the attribute. ## DOM properties are typed @@ -216,9 +216,9 @@ There are other examples. The `style` attribute is a string, but the `style` pro </script> ``` -That's an important difference. But even if a DOM property type is a string, it may differ from the attribute! +Most properties are strings though. -For instance, the `href` DOM property is always a *full* URL, even if the attribute contains a relative URL or just a `#hash`. +Quite rarely, even if a DOM property type is a string, it may differ from the attribute. For instance, the `href` DOM property is always a *full* URL, even if the attribute contains a relative URL or just a `#hash`. Here's an example: @@ -260,7 +260,7 @@ Like this: for(let div of document.querySelectorAll('[show-info]')) { // insert the corresponding info into the field let field = div.getAttribute('show-info'); - div.innerHTML = user[field]; // Pete, then age + div.innerHTML = user[field]; // first Pete into "name", then 25 into "age" } </script> ``` @@ -298,16 +298,16 @@ For instance, here for the order state the attribute `order-state` is used: </div> ``` -Why the attribute may be preferable to classes like `.order-state-new`, `.order-state-pending`, `order-state-canceled`? +Why would using an attribute be preferable to having classes like `.order-state-new`, `.order-state-pending`, `.order-state-canceled`? -That's because an attribute is more convenient to manage. The state can be changed as easy as: +Because an attribute is more convenient to manage. The state can be changed as easy as: ```js // a bit simpler than removing old/adding a new class div.setAttribute('order-state', 'canceled'); ``` -But there may be a possible problem with custom attributes. What if we use a non-standard attribute for our purposes and later the standard introduces it and makes it do something? The HTML language is alive, it grows, more attributes appear to suit the needs of developers. There may be unexpected effects in such case. +But there may be a possible problem with custom attributes. What if we use a non-standard attribute for our purposes and later the standard introduces it and makes it do something? The HTML language is alive, it grows, and more attributes appear to suit the needs of developers. There may be unexpected effects in such case. To avoid conflicts, there exist [data-*](https://html.spec.whatwg.org/#embedding-custom-non-visible-data-with-the-data-*-attributes) attributes. @@ -380,7 +380,7 @@ Methods to work with attributes are: - `elem.removeAttribute(name)` -- to remove the attribute. - `elem.attributes` is a collection of all attributes. -For most needs, DOM properties can serve us well. We should refer to attributes only when DOM properties do not suit us, when we need exactly attributes, for instance: +For most situations using DOM properties is preferable. We should refer to attributes only when DOM properties do not suit us, when we need exactly attributes, for instance: - We need a non-standard attribute. But if it starts with `data-`, then we should use `dataset`. - We want to read the value "as written" in HTML. The value of the DOM property may be different, for instance the `href` property is always a full URL, and we may want to get the "original" value. diff --git a/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/solution.md b/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/solution.md index 93ae862fd..a38f01645 100644 --- a/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/solution.md +++ b/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/solution.md @@ -12,7 +12,7 @@ Here's an example: let text = '<b>text</b>'; elem1.append(document.createTextNode(text)); - elem2.textContent = text; - elem3.innerHTML = text; + elem2.innerHTML = text; + elem3.textContent = text; </script> ``` diff --git a/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/task.md b/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/task.md index e127bc0ef..40c75dff3 100644 --- a/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/task.md +++ b/2-ui/1-document/07-modifying-document/1-createtextnode-vs-innerhtml/task.md @@ -6,7 +6,7 @@ importance: 5 We have an empty DOM element `elem` and a string `text`. -Which of these 3 commands do exactly the same? +Which of these 3 commands will do exactly the same? 1. `elem.append(document.createTextNode(text))` 2. `elem.innerHTML = text` diff --git a/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.md b/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.md index 15238fcf4..1414e90c1 100644 --- a/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.md +++ b/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.md @@ -39,15 +39,19 @@ The clock-managing functions: ```js let timerId; -function clockStart() { // run the clock - timerId = setInterval(update, 1000); +function clockStart() { // run the clock + if (!timerId) { // only set a new interval if the clock is not running + timerId = setInterval(update, 1000); + } update(); // (*) } function clockStop() { clearInterval(timerId); - timerId = null; + timerId = null; // (**) } ``` Please note that the call to `update()` is not only scheduled in `clockStart()`, but immediately run in the line `(*)`. Otherwise the visitor would have to wait till the first execution of `setInterval`. And the clock would be empty till then. + +Also it is important to set a new interval in `clockStart()` only when the clock is not running. Otherways clicking the start button several times would set multiple concurrent intervals. Even worse - we would only keep the `timerID` of the last interval, losing references to all others. Then we wouldn't be able to stop the clock ever again! Note that we need to clear the `timerID` when the clock is stopped in the line `(**)`, so that it can be started again by running `clockStart()`. diff --git a/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.view/index.html b/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.view/index.html index 1bf642b10..84ee26f19 100644 --- a/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.view/index.html +++ b/2-ui/1-document/07-modifying-document/10-clock-setinterval/solution.view/index.html @@ -43,15 +43,19 @@ } function clockStart() { - timerId = setInterval(update, 1000); + // set a new interval only if the clock is stopped + // otherwise we would rewrite the timerID reference to the running interval and wouldn't be able to stop the clock ever again + if (!timerId) { + timerId = setInterval(update, 1000); + } update(); // <-- start right now, don't wait 1 second till the first setInterval works } function clockStop() { clearInterval(timerId); + timerId = null; // <-- clear timerID to indicate that the clock has been stopped, so that it is possible to start it again in clockStart() } - clockStart(); </script> <!-- click on this button calls clockStart() --> diff --git a/2-ui/1-document/07-modifying-document/12-sort-table/solution.md b/2-ui/1-document/07-modifying-document/12-sort-table/solution.md index 25dd58b0c..49243e8e3 100644 --- a/2-ui/1-document/07-modifying-document/12-sort-table/solution.md +++ b/2-ui/1-document/07-modifying-document/12-sort-table/solution.md @@ -1,19 +1,18 @@ The solution is short, yet may look a bit tricky, so here I provide it with extensive comments: - ```js -let sortedRows = Array.from(table.rows) - .slice(1) - .sort((rowA, rowB) => rowA.cells[0].innerHTML > rowB.cells[0].innerHTML ? 1 : -1); +let sortedRows = Array.from(table.tBodies[0].rows) // 1 + .sort((rowA, rowB) => rowA.cells[0].innerHTML.localeCompare(rowB.cells[0].innerHTML)); -table.tBodies[0].append(...sortedRows); +table.tBodies[0].append(...sortedRows); // (3) ``` -1. Get all `<tr>`, like `table.querySelectorAll('tr')`, then make an array from them, cause we need array methods. -2. The first TR (`table.rows[0]`) is actually a table header, so we take the rest by `.slice(1)`. -3. Then sort them comparing by the content of the first `<td>` (the name field). -4. Now insert nodes in the right order by `.append(...sortedRows)`. +The step-by-step algorthm: + +1. Get all `<tr>`, from `<tbody>`. +2. Then sort them comparing by the content of the first `<td>` (the name field). +3. Now insert nodes in the right order by `.append(...sortedRows)`. - Tables always have an implicit <tbody> element, so we need to take it and insert into it: a simple `table.append(...)` would fail. +We don't have to remove row elements, just "re-insert", they leave the old place automatically. - Please note: we don't have to remove them, just "re-insert", they leave the old place automatically. +P.S. In our case, there's an explicit `<tbody>` in the table, but even if HTML table doesn't have `<tbody>`, the DOM structure always has it. diff --git a/2-ui/1-document/07-modifying-document/12-sort-table/solution.view/index.html b/2-ui/1-document/07-modifying-document/12-sort-table/solution.view/index.html index 81e985748..40692031a 100644 --- a/2-ui/1-document/07-modifying-document/12-sort-table/solution.view/index.html +++ b/2-ui/1-document/07-modifying-document/12-sort-table/solution.view/index.html @@ -1,37 +1,30 @@ <!DOCTYPE html> -<html> -<body> - <table id="table"> - <tr> - <th>Name</th> - <th>Surname</th> - <th>Age</th> - </tr> - <tr> - <td>John</td> - <td>Smith</td> - <td>10</td> - </tr> - <tr> - <td>Pete</td> - <td>Brown</td> - <td>15</td> - </tr> - <tr> - <td>Ann</td> - <td>Lee</td> - <td>5</td> - </tr> - </table> +<table id="table"> +<thead> + <tr> + <th>Name</th><th>Surname</th><th>Age</th> + </tr> +</thead> +<tbody> + <tr> + <td>John</td><td>Smith</td><td>10</td> + </tr> + <tr> + <td>Pete</td><td>Brown</td><td>15</td> + </tr> + <tr> + <td>Ann</td><td>Lee</td><td>5</td> + </tr> + <tr> + <td>...</td><td>...</td><td>...</td> + </tr> +</tbody> +</table> - <script> - let sortedRows = Array.from(table.rows) - .slice(1) - .sort((rowA, rowB) => rowA.cells[0].innerHTML > rowB.cells[0].innerHTML ? 1 : -1); +<script> + let sortedRows = Array.from(table.tBodies[0].rows) + .sort((rowA, rowB) => rowA.cells[0].innerHTML.localeCompare(rowB.cells[0].innerHTML)); - table.tBodies[0].append(...sortedRows); - </script> - -</body> -</html> + table.tBodies[0].append(...sortedRows); +</script> diff --git a/2-ui/1-document/07-modifying-document/12-sort-table/source.view/index.html b/2-ui/1-document/07-modifying-document/12-sort-table/source.view/index.html index e41eb229f..9071c88ee 100644 --- a/2-ui/1-document/07-modifying-document/12-sort-table/source.view/index.html +++ b/2-ui/1-document/07-modifying-document/12-sort-table/source.view/index.html @@ -1,33 +1,27 @@ <!DOCTYPE html> -<html> -<body> - <table id="table"> - <tr> - <th>Name</th> - <th>Surname</th> - <th>Age</th> - </tr> - <tr> - <td>John</td> - <td>Smith</td> - <td>10</td> - </tr> - <tr> - <td>Pete</td> - <td>Brown</td> - <td>15</td> - </tr> - <tr> - <td>Ann</td> - <td>Lee</td> - <td>5</td> - </tr> - </table> +<table id="table"> +<thead> + <tr> + <th>Name</th><th>Surname</th><th>Age</th> + </tr> +</thead> +<tbody> + <tr> + <td>John</td><td>Smith</td><td>10</td> + </tr> + <tr> + <td>Pete</td><td>Brown</td><td>15</td> + </tr> + <tr> + <td>Ann</td><td>Lee</td><td>5</td> + </tr> + <tr> + <td>...</td><td>...</td><td>...</td> + </tr> +</tbody> +</table> - <script> - // ... your code ... - </script> - -</body> -</html> +<script> + // ... your code ... +</script> diff --git a/2-ui/1-document/07-modifying-document/12-sort-table/task.md b/2-ui/1-document/07-modifying-document/12-sort-table/task.md index 41d6fca29..7cdba35bc 100644 --- a/2-ui/1-document/07-modifying-document/12-sort-table/task.md +++ b/2-ui/1-document/07-modifying-document/12-sort-table/task.md @@ -6,33 +6,29 @@ importance: 5 There's a table: +```html run <table> -<tr> - <th>Name</th> - <th>Surname</th> - <th>Age</th> -</tr> -<tr> - <td>John</td> - <td>Smith</td> - <td>10</td> -</tr> -<tr> - <td>Pete</td> - <td>Brown</td> - <td>15</td> -</tr> -<tr> - <td>Ann</td> - <td>Lee</td> - <td>5</td> -</tr> -<tr> - <td>...</td> - <td>...</td> - <td>...</td> -</tr> +<thead> + <tr> + <th>Name</th><th>Surname</th><th>Age</th> + </tr> +</thead> +<tbody> + <tr> + <td>John</td><td>Smith</td><td>10</td> + </tr> + <tr> + <td>Pete</td><td>Brown</td><td>15</td> + </tr> + <tr> + <td>Ann</td><td>Lee</td><td>5</td> + </tr> + <tr> + <td>...</td><td>...</td><td>...</td> + </tr> +</tbody> </table> +``` There may be more rows in it. diff --git a/2-ui/1-document/07-modifying-document/5-why-aaa/solution.md b/2-ui/1-document/07-modifying-document/5-why-aaa/solution.md index 6b85168b9..3d1f6698f 100644 --- a/2-ui/1-document/07-modifying-document/5-why-aaa/solution.md +++ b/2-ui/1-document/07-modifying-document/5-why-aaa/solution.md @@ -1,9 +1,9 @@ The HTML in the task is incorrect. That's the reason of the odd thing. -The browser has to fix it automatically. But there may be no text inside the `<table>`: according to the spec only table-specific tags are allowed. So the browser adds `"aaa"` *before* the `<table>`. +The browser has to fix it automatically. But there may be no text inside the `<table>`: according to the spec only table-specific tags are allowed. So the browser shows `"aaa"` *before* the `<table>`. Now it's obvious that when we remove the table, it remains. -The question can be easily answered by exploring the DOM using the browser tools. It shows `"aaa"` before the `<table>`. +The question can be easily answered by exploring the DOM using the browser tools. You'll see `"aaa"` before the `<table>`. The HTML standard specifies in detail how to process bad HTML, and such behavior of the browser is correct. diff --git a/2-ui/1-document/07-modifying-document/5-why-aaa/task.md b/2-ui/1-document/07-modifying-document/5-why-aaa/task.md index 03064ed2c..861f70503 100644 --- a/2-ui/1-document/07-modifying-document/5-why-aaa/task.md +++ b/2-ui/1-document/07-modifying-document/5-why-aaa/task.md @@ -4,7 +4,11 @@ importance: 1 # Why does "aaa" remain? -Run the example. Why does `table.remove()` not delete the text `"aaa"`? +In the example below, the call `table.remove()` removes the table from the document. + +But if you run it, you can see that the text `"aaa"` is still visible. + +Why does that happen? ```html height=100 run <table id="table"> @@ -18,6 +22,6 @@ Run the example. Why does `table.remove()` not delete the text `"aaa"`? alert(table); // the table, as it should be table.remove(); - // why there's still aaa in the document? + // why there's still "aaa" in the document? </script> ``` diff --git a/2-ui/1-document/07-modifying-document/6-create-list/task.md b/2-ui/1-document/07-modifying-document/6-create-list/task.md index 43b0a34a7..a57e7e2d9 100644 --- a/2-ui/1-document/07-modifying-document/6-create-list/task.md +++ b/2-ui/1-document/07-modifying-document/6-create-list/task.md @@ -10,7 +10,7 @@ For every list item: 1. Ask a user about its content using `prompt`. 2. Create the `<li>` with it and add it to `<ul>`. -3. Continue until the user cancels the input (by pressing `key:Esc` or CANCEL in prompt). +3. Continue until the user cancels the input (by pressing `key:Esc` or via an empty entry). All elements should be created dynamically. diff --git a/2-ui/1-document/07-modifying-document/7-create-object-tree/build-tree-dom.view/index.html b/2-ui/1-document/07-modifying-document/7-create-object-tree/build-tree-dom.view/index.html index aec462fe8..06d9c01b1 100755 --- a/2-ui/1-document/07-modifying-document/7-create-object-tree/build-tree-dom.view/index.html +++ b/2-ui/1-document/07-modifying-document/7-create-object-tree/build-tree-dom.view/index.html @@ -17,7 +17,7 @@ "oak": {} }, "Flowering": { - "redbud": {}, + "apple tree": {}, "magnolia": {} } } diff --git a/2-ui/1-document/07-modifying-document/7-create-object-tree/innerhtml.view/index.html b/2-ui/1-document/07-modifying-document/7-create-object-tree/innerhtml.view/index.html index fd85e6c76..0f5f6b037 100644 --- a/2-ui/1-document/07-modifying-document/7-create-object-tree/innerhtml.view/index.html +++ b/2-ui/1-document/07-modifying-document/7-create-object-tree/innerhtml.view/index.html @@ -17,7 +17,7 @@ "oak": {} }, "Flowering": { - "redbud": {}, + "apple tree": {}, "magnolia": {} } } diff --git a/2-ui/1-document/07-modifying-document/7-create-object-tree/source.view/index.html b/2-ui/1-document/07-modifying-document/7-create-object-tree/source.view/index.html index d035aec68..8586f6b24 100755 --- a/2-ui/1-document/07-modifying-document/7-create-object-tree/source.view/index.html +++ b/2-ui/1-document/07-modifying-document/7-create-object-tree/source.view/index.html @@ -28,7 +28,7 @@ </li> <li>Flowering <ul> - <li>redbud</li> + <li>apple tree</li> <li>magnolia</li> </ul> </li> @@ -51,7 +51,7 @@ "oak": {} }, "Flowering": { - "redbud": {}, + "apple tree": {}, "magnolia": {} } } diff --git a/2-ui/1-document/07-modifying-document/7-create-object-tree/task.md b/2-ui/1-document/07-modifying-document/7-create-object-tree/task.md index 6b60c096f..5ec1a01bc 100644 --- a/2-ui/1-document/07-modifying-document/7-create-object-tree/task.md +++ b/2-ui/1-document/07-modifying-document/7-create-object-tree/task.md @@ -21,7 +21,7 @@ let data = { "oak": {} }, "Flowering": { - "redbud": {}, + "apple tree": {}, "magnolia": {} } } diff --git a/2-ui/1-document/07-modifying-document/9-calendar-table/solution.md b/2-ui/1-document/07-modifying-document/9-calendar-table/solution.md index 67bb5e13d..de8be56e9 100644 --- a/2-ui/1-document/07-modifying-document/9-calendar-table/solution.md +++ b/2-ui/1-document/07-modifying-document/9-calendar-table/solution.md @@ -3,7 +3,7 @@ We'll create the table as a string: `"<table>...</table>"`, and then assign it t The algorithm: 1. Create the table header with `<th>` and weekday names. -1. Create the date object `d = new Date(year, month-1)`. That's the first day of `month` (taking into account that months in JavaScript start from `0`, not `1`). -2. First few cells till the first day of the month `d.getDay()` may be empty. Let's fill them in with `<td></td>`. -3. Increase the day in `d`: `d.setDate(d.getDate()+1)`. If `d.getMonth()` is not yet the next month, then add the new cell `<td>` to the calendar. If that's a Sunday, then add a newline <code>"</tr><tr>"</code>. -4. If the month has finished, but the table row is not yet full, add empty `<td>` into it, to make it square. +2. Create the date object `d = new Date(year, month-1)`. That's the first day of `month` (taking into account that months in JavaScript start from `0`, not `1`). +3. First few cells till the first day of the month `d.getDay()` may be empty. Let's fill them in with `<td></td>`. +4. Increase the day in `d`: `d.setDate(d.getDate()+1)`. If `d.getMonth()` is not yet the next month, then add the new cell `<td>` to the calendar. If that's a Sunday, then add a newline <code>"</tr><tr>"</code>. +5. If the month has finished, but the table row is not yet full, add empty `<td>` into it, to make it square. diff --git a/2-ui/1-document/07-modifying-document/article.md b/2-ui/1-document/07-modifying-document/article.md index 8713e1850..75ce1fbb0 100644 --- a/2-ui/1-document/07-modifying-document/article.md +++ b/2-ui/1-document/07-modifying-document/article.md @@ -1,14 +1,12 @@ # Modifying the document -DOM modifications is the key to create "live" pages. +DOM modification is the key to creating "live" pages. Here we'll see how to create new elements "on the fly" and modify the existing page content. -First we'll see a simple example and then explain the methods. - ## Example: show a message -For a start, let's see how to add a message on the page that looks nicer than `alert`. +Let's demonstrate using an example. We'll add a message on the page that looks nicer than `alert`. Here's how it will look: @@ -30,15 +28,14 @@ Here's how it will look: */!* ``` -That was an HTML example. Now let's create the same `div` with JavaScript (assuming that the styles are still in the HTML or an external CSS file). +That was the HTML example. Now let's create the same `div` with JavaScript (assuming that the styles are in the HTML/CSS already). ## Creating an element - To create DOM nodes, there are two methods: `document.createElement(tag)` -: Creates a new element with the given tag: +: Creates a new *element node* with the given tag: ```js let div = document.createElement('div'); @@ -51,23 +48,30 @@ To create DOM nodes, there are two methods: let textNode = document.createTextNode('Here I am'); ``` +Most of the time we need to create element nodes, such as the `div` for the message. + ### Creating the message -In our case we want to make a `div` with given classes and the message in it: +Creating the message div takes 3 steps: ```js +// 1. Create <div> element let div = document.createElement('div'); -div.className = "alert alert-success"; + +// 2. Set its class to "alert" +div.className = "alert"; + +// 3. Fill it with the content div.innerHTML = "<strong>Hi there!</strong> You've read an important message."; ``` -After that, we have our DOM element ready. Right now it is just in a variable and we cannot see it. That is because it's not yet inserted into the page. +We've created the element. But as of now it's only in a variable named `div`, not in the page yet. So we can't see it. ## Insertion methods -To make the `div` show up, we need to insert it somewhere into `document`. For instance, in `document.body`. +To make the `div` show up, we need to insert it somewhere into `document`. For instance, into `<body>` element, referenced by `document.body`. -There's a special method for that: `document.body.appendChild(div)`. +There's a special method `append` for that: `document.body.append(div)`. Here's the full code: @@ -84,85 +88,30 @@ Here's the full code: <script> let div = document.createElement('div'); - div.className = "alert alert-success"; + div.className = "alert"; div.innerHTML = "<strong>Hi there!</strong> You've read an important message."; *!* - document.body.appendChild(div); + document.body.append(div); */!* </script> ``` -Here's a brief list of methods to insert a node into a parent element (`parentElem` for short): - -`parentElem.appendChild(node)` -: Appends `node` as the last child of `parentElem`. - - The following example adds a new `<li>` to the end of `<ol>`: - - ```html run height=100 - <ol id="list"> - <li>0</li> - <li>1</li> - <li>2</li> - </ol> +Here we called `append` on `document.body`, but we can call `append` method on any other element, to put another element into it. For instance, we can append something to `<div>` by calling `div.append(anotherElement)`. - <script> - let newLi = document.createElement('li'); - newLi.innerHTML = 'Hello, world!'; - - list.appendChild(newLi); - </script> - ``` - -`parentElem.insertBefore(node, nextSibling)` -: Inserts `node` before `nextSibling` into `parentElem`. - - The following code inserts a new list item before the second `<li>`: - - ```html run height=100 - <ol id="list"> - <li>0</li> - <li>1</li> - <li>2</li> - </ol> - <script> - let newLi = document.createElement('li'); - newLi.innerHTML = 'Hello, world!'; - - *!* - list.insertBefore(newLi, list.children[1]); - */!* - </script> - ``` - To insert `newLi` as the first element, we can do it like this: +Here are more insertion methods, they specify different places where to insert: - ```js - list.insertBefore(newLi, list.firstChild); - ``` - -`parentElem.replaceChild(node, oldChild)` -: Replaces `oldChild` with `node` among children of `parentElem`. - -All these methods return the inserted node. In other words, `parentElem.appendChild(node)` returns `node`. But usually the returned value is not used, we just run the method. - -These methods are "old school": they exist from the ancient times and we can meet them in many old scripts. Unfortunately, there are some tasks that are hard to solve with them. - -For instance, how to insert *html* if we have it as a string? Or, given a node, how to insert another node *before* it? Of course, all that is doable, but not in an elegant way. - -So there exist two other sets of insertion methods to handle all cases easily. +- `node.append(...nodes or strings)` -- append nodes or strings *at the end* of `node`, +- `node.prepend(...nodes or strings)` -- insert nodes or strings *at the beginning* of `node`, +- `node.before(...nodes or strings)` –- insert nodes or strings *before* `node`, +- `node.after(...nodes or strings)` –- insert nodes or strings *after* `node`, +- `node.replaceWith(...nodes or strings)` –- replaces `node` with the given nodes or strings. -### prepend/append/before/after +Arguments of these methods are an arbitrary list of DOM nodes to insert, or text strings (that become text nodes automatically). -This set of methods provides more flexible insertions: +Let's see them in action. -- `node.append(...nodes or strings)` -- append nodes or strings at the end of `node`, -- `node.prepend(...nodes or strings)` -- insert nodes or strings into the beginning of `node`, -- `node.before(...nodes or strings)` –- insert nodes or strings before the `node`, -- `node.after(...nodes or strings)` –- insert nodes or strings after the `node`, -- `node.replaceWith(...nodes or strings)` –- replaces `node` with the given nodes or strings. - -Here's an example of using these methods to add more items to a list and the text before/after it: +Here's an example of using these methods to add items to a list and the text before/after it: ```html autorun <ol id="ol"> @@ -172,20 +121,20 @@ Here's an example of using these methods to add more items to a list and the tex </ol> <script> - ol.before('before'); - ol.after('after'); + ol.before('before'); // insert string "before" before <ol> + ol.after('after'); // insert string "after" after <ol> - let prepend = document.createElement('li'); - prepend.innerHTML = 'prepend'; - ol.prepend(prepend); + let liFirst = document.createElement('li'); + liFirst.innerHTML = 'prepend'; + ol.prepend(liFirst); // insert liFirst at the beginning of <ol> - let append = document.createElement('li'); - append.innerHTML = 'append'; - ol.append(append); + let liLast = document.createElement('li'); + liLast.innerHTML = 'append'; + ol.append(liLast); // insert liLast at the end of <ol> </script> ``` -Here's a small picture what methods do: +Here's a visual picture of what the methods do:  @@ -203,7 +152,7 @@ before after ``` -These methods can insert multiple lists of nodes and text pieces in a single call. +As said, these methods can insert multiple nodes and text pieces in a single call. For instance, here a string and an element are inserted: @@ -214,7 +163,7 @@ For instance, here a string and an element are inserted: </script> ``` -All text is inserted *as text*. +Please note: the text is inserted "as text", not "as HTML", with proper escaping of characters such as `<`, `>`. So the final HTML is: @@ -230,20 +179,20 @@ In other words, strings are inserted in a safe way, like `elem.textContent` does So, these methods can only be used to insert DOM nodes or text pieces. -But what if we want to insert HTML "as html", with all tags and stuff working, like `elem.innerHTML`? +But what if we'd like to insert an HTML string "as html", with all tags and stuff working, in the same manner as `elem.innerHTML` does it? -### insertAdjacentHTML/Text/Element +## insertAdjacentHTML/Text/Element -There's another, pretty versatile method: `elem.insertAdjacentHTML(where, html)`. +For that we can use another, pretty versatile method: `elem.insertAdjacentHTML(where, html)`. -The first parameter is a string, specifying where to insert. Must be one of the following: +The first parameter is a code word, specifying where to insert relative to `elem`. Must be one of the following: -- `"beforebegin"` -- insert `html` before `elem`, +- `"beforebegin"` -- insert `html` immediately before `elem`, - `"afterbegin"` -- insert `html` into `elem`, at the beginning, - `"beforeend"` -- insert `html` into `elem`, at the end, -- `"afterend"` -- insert `html` after `elem`. +- `"afterend"` -- insert `html` immediately after `elem`. -The second parameter is an HTML string, inserted "as is". +The second parameter is an HTML string, that is inserted "as HTML". For instance: @@ -263,7 +212,7 @@ For instance: <p>Bye</p> ``` -That's how we can append an arbitrary HTML to our page. +That's how we can append arbitrary HTML to the page. Here's the picture of insertion variants: @@ -292,12 +241,56 @@ So here's an alternative variant of showing a message: </style> <script> - document.body.insertAdjacentHTML("afterbegin", `<div class="alert alert-success"> + document.body.insertAdjacentHTML("afterbegin", `<div class="alert"> <strong>Hi there!</strong> You've read an important message. </div>`); </script> ``` +## Node removal + +To remove a node, there's a method `node.remove()`. + +Let's make our message disappear after a second: + +```html run untrusted +<style> +.alert { + padding: 15px; + border: 1px solid #d6e9c6; + border-radius: 4px; + color: #3c763d; + background-color: #dff0d8; +} +</style> + +<script> + let div = document.createElement('div'); + div.className = "alert"; + div.innerHTML = "<strong>Hi there!</strong> You've read an important message."; + + document.body.append(div); +*!* + setTimeout(() => div.remove(), 1000); +*/!* +</script> +``` + +Please note: if we want to *move* an element to another place -- there's no need to remove it from the old one. + +**All insertion methods automatically remove the node from the old place.** + +For instance, let's swap elements: + +```html run height=50 +<div id="first">First</div> +<div id="second">Second</div> +<script> + // no need to call remove + second.after(first); // take #second and after it insert #first +</script> +``` + ## Cloning nodes: cloneNode How to insert one more similar message? @@ -335,12 +328,11 @@ An example of copying the message: </script> ``` - ## DocumentFragment [#document-fragment] -`DocumentFragment` is a special DOM node that serves as a wrapper to pass around groups of nodes. +`DocumentFragment` is a special DOM node that serves as a wrapper to pass around lists of nodes. -We can append other nodes to it, but when we insert it somewhere, then it "disappears", leaving its content inserted instead. +We can append other nodes to it, but when we insert it somewhere, then its content is inserted instead. For example, `getListContent` below generates a fragment with `<li>` items, that are later inserted into `<ul>`: @@ -402,62 +394,84 @@ ul.append(...getListContent()); // append + "..." operator = friends! We mention `DocumentFragment` mainly because there are some concepts on top of it, like [template](info:template-element) element, that we'll cover much later. +## Old-school insert/remove methods -## Removal methods +[old] -To remove nodes, there are the following methods: +There are also "old school" DOM manipulation methods, existing for historical reasons. +These methods come from really ancient times. Nowadays, there's no reason to use them, as modern methods, such as `append`, `prepend`, `before`, `after`, `remove`, `replaceWith`, are more flexible. -`parentElem.removeChild(node)` -: Removes `node` from `parentElem` (assuming it's a child). +The only reason we list these methods here is that you can find them in many old scripts: -`node.remove()` -: Removes the `node` from its place. +`parentElem.appendChild(node)` +: Appends `node` as the last child of `parentElem`. -We can easily see that the second method is much shorter. The first one exists for historical reasons. + The following example adds a new `<li>` to the end of `<ol>`: -````smart -If we want to *move* an element to another place -- there's no need to remove it from the old one. + ```html run height=100 + <ol id="list"> + <li>0</li> + <li>1</li> + <li>2</li> + </ol> -**All insertion methods automatically remove the node from the old place.** + <script> + let newLi = document.createElement('li'); + newLi.innerHTML = 'Hello, world!'; -For instance, let's swap elements: + list.appendChild(newLi); + </script> + ``` -```html run height=50 -<div id="first">First</div> -<div id="second">Second</div> -<script> - // no need to call remove - second.after(first); // take #second and after it - insert #first -</script> -``` -```` +`parentElem.insertBefore(node, nextSibling)` +: Inserts `node` before `nextSibling` into `parentElem`. -Let's make our message disappear after a second: + The following code inserts a new list item before the second `<li>`: -```html run untrusted -<style> -.alert { - padding: 15px; - border: 1px solid #d6e9c6; - border-radius: 4px; - color: #3c763d; - background-color: #dff0d8; -} -</style> + ```html run height=100 + <ol id="list"> + <li>0</li> + <li>1</li> + <li>2</li> + </ol> + <script> + let newLi = document.createElement('li'); + newLi.innerHTML = 'Hello, world!'; -<script> - let div = document.createElement('div'); - div.className = "alert alert-success"; - div.innerHTML = "<strong>Hi there!</strong> You've read an important message."; + *!* + list.insertBefore(newLi, list.children[1]); + */!* + </script> + ``` + To insert `newLi` as the first element, we can do it like this: - document.body.append(div); -*!* - setTimeout(() => div.remove(), 1000); - // or setTimeout(() => document.body.removeChild(div), 1000); -*/!* -</script> -``` + ```js + list.insertBefore(newLi, list.firstChild); + ``` + +`parentElem.replaceChild(node, oldChild)` +: Replaces `oldChild` with `node` among children of `parentElem`. + +`parentElem.removeChild(node)` +: Removes `node` from `parentElem` (assuming `node` is its child). + + The following example removes first `<li>` from `<ol>`: + + ```html run height=100 + <ol id="list"> + <li>0</li> + <li>1</li> + <li>2</li> + </ol> + + <script> + let li = list.firstElementChild; + list.removeChild(li); + </script> + ``` + +All these methods return the inserted/removed node. In other words, `parentElem.appendChild(node)` returns `node`. But usually the returned value is not used, we just run the method. ## A word about "document.write" @@ -500,51 +514,48 @@ For instance: So it's kind of unusable at "after loaded" stage, unlike other DOM methods we covered above. -That was the downside. +That's the downside. -Technically, when `document.write` is called while the browser is still reading HTML, it appends something to it, and the browser consumes it just as it were initially there. +There's an upside also. Technically, when `document.write` is called while the browser is reading ("parsing") incoming HTML, and it writes something, the browser consumes it just as if it were initially there, in the HTML text. -That gives us the upside -- it works blazingly fast, because there's *no DOM modification*. It writes directly into the page text, while the DOM is not yet built, and the browser puts it into DOM at generation-time. +So it works blazingly fast, because there's *no DOM modification* involved. It writes directly into the page text, while the DOM is not yet built. So if we need to add a lot of text into HTML dynamically, and we're at page loading phase, and the speed matters, it may help. But in practice these requirements rarely come together. And usually we can see this method in scripts just because they are old. ## Summary -Methods to create new nodes: - -- `document.createElement(tag)` -- creates an element with the given tag, -- `document.createTextNode(value)` -- creates a text node (rarely used), -- `elem.cloneNode(deep)` -- clones the element, if `deep==true` then with all descendants. - -Insertion and removal of nodes: +- Methods to create new nodes: + - `document.createElement(tag)` -- creates an element with the given tag, + - `document.createTextNode(value)` -- creates a text node (rarely used), + - `elem.cloneNode(deep)` -- clones the element, if `deep==true` then with all descendants. -- From the parent: - - `parent.appendChild(node)` - - `parent.insertBefore(node, nextSibling)` - - `parent.removeChild(node)` - - `parent.replaceChild(newElem, node)` +- Insertion and removal: + - `node.append(...nodes or strings)` -- insert into `node`, at the end, + - `node.prepend(...nodes or strings)` -- insert into `node`, at the beginning, + - `node.before(...nodes or strings)` –- insert right before `node`, + - `node.after(...nodes or strings)` –- insert right after `node`, + - `node.replaceWith(...nodes or strings)` –- replace `node`. + - `node.remove()` –- remove the `node`. - All these methods return `node`. + Text strings are inserted "as text". -- Given a list of nodes and strings: - - `node.append(...nodes or strings)` -- insert into `node`, at the end, - - `node.prepend(...nodes or strings)` -- insert into `node`, at the beginning, - - `node.before(...nodes or strings)` –- insert right before `node`, - - `node.after(...nodes or strings)` –- insert right after `node`, - - `node.replaceWith(...nodes or strings)` –- replace `node`. - - `node.remove()` –- remove the `node`. +- There are also "old school" methods: + - `parent.appendChild(node)` + - `parent.insertBefore(node, nextSibling)` + - `parent.removeChild(node)` + - `parent.replaceChild(newElem, node)` - Text strings are inserted "as text". + All these methods return `node`. -- Given a piece of HTML: `elem.insertAdjacentHTML(where, html)`, inserts depending on where: - - `"beforebegin"` -- insert `html` right before `elem`, - - `"afterbegin"` -- insert `html` into `elem`, at the beginning, - - `"beforeend"` -- insert `html` into `elem`, at the end, - - `"afterend"` -- insert `html` right after `elem`. +- Given some HTML in `html`, `elem.insertAdjacentHTML(where, html)` inserts it depending on the value of `where`: + - `"beforebegin"` -- insert `html` right before `elem`, + - `"afterbegin"` -- insert `html` into `elem`, at the beginning, + - `"beforeend"` -- insert `html` into `elem`, at the end, + - `"afterend"` -- insert `html` right after `elem`. - Also there are similar methods `elem.insertAdjacentText` and `elem.insertAdjacentElement`, they insert text strings and elements, but they are rarely used. + Also there are similar methods, `elem.insertAdjacentText` and `elem.insertAdjacentElement`, that insert text strings and elements, but they are rarely used. - To append HTML to the page before it has finished loading: - - `document.write(html)` + - `document.write(html)` - After the page is loaded such a call erases the document. Mostly seen in old scripts. + After the page is loaded such a call erases the document. Mostly seen in old scripts. diff --git a/2-ui/1-document/08-styles-and-classes/article.md b/2-ui/1-document/08-styles-and-classes/article.md index 5159c532d..46aaa3b00 100644 --- a/2-ui/1-document/08-styles-and-classes/article.md +++ b/2-ui/1-document/08-styles-and-classes/article.md @@ -7,20 +7,21 @@ There are generally two ways to style an element: 1. Create a class in CSS and add it: `<div class="...">` 2. Write properties directly into `style`: `<div style="...">`. -CSS is always the preferred way -- not only for HTML, but in JavaScript as well. +JavaScript can modify both classes and `style` properties. -We should only manipulate the `style` property if classes "can't handle it". +We should always prefer CSS classes to `style`. The latter should only be used if classes "can't handle it". -For instance, `style` is acceptable if we calculate coordinates of an element dynamically and want to set them from JavaScript, like this: +For example, `style` is acceptable if we calculate coordinates of an element dynamically and want to set them from JavaScript, like this: ```js let top = /* complex calculations */; let left = /* complex calculations */; -elem.style.left = left; // e.g '123px' + +elem.style.left = left; // e.g '123px', calculated at run-time elem.style.top = top; // e.g '456px' ``` -For other cases, like making the text red, adding a background icon -- describe that in CSS and then apply the class. That's more flexible and easier to support. +For other cases, like making the text red, adding a background icon -- describe that in CSS and then add the class (JavaScript can do that). That's more flexible and easier to support. ## className and classList @@ -40,11 +41,11 @@ For instance: </body> ``` -If we assign something to `elem.className`, it replaces the whole strings of classes. Sometimes that's what we need, but often we want to add/remove a single class. +If we assign something to `elem.className`, it replaces the whole string of classes. Sometimes that's what we need, but often we want to add/remove a single class. There's another property for that: `elem.classList`. -The `elem.classList` is a special object with methods to `add/remove/toggle` classes. +The `elem.classList` is a special object with methods to `add/remove/toggle` a single class. For instance: @@ -66,10 +67,10 @@ So we can operate both on the full class string using `className` or on individu Methods of `classList`: - `elem.classList.add/remove("class")` -- adds/removes the class. -- `elem.classList.toggle("class")` -- if the class exists, then removes it, otherwise adds it. -- `elem.classList.contains("class")` -- returns `true/false`, checks for the given class. +- `elem.classList.toggle("class")` -- adds the class if it doesn't exist, otherwise removes it. +- `elem.classList.contains("class")` -- checks for the given class, returns `true/false`. -Besides that, `classList` is iterable, so we can list all classes like this: +Besides, `classList` is iterable, so we can list all classes with `for..of`, like this: ```html run <body class="main page"> @@ -83,7 +84,7 @@ Besides that, `classList` is iterable, so we can list all classes like this: ## Element style -The property `elem.style` is an object that corresponds to what's written in the `"style"` attribute. Setting `elem.style.width="100px"` works as if we had in the attribute `style="width:100px"`. +The property `elem.style` is an object that corresponds to what's written in the `"style"` attribute. Setting `elem.style.width="100px"` works the same as if we had in the attribute `style` a string `width:100px`. For multi-word property the camelCase is used: @@ -100,14 +101,14 @@ document.body.style.backgroundColor = prompt('background color?', 'green'); ``` ````smart header="Prefixed properties" -Browser-prefixed properties like `-moz-border-radius`, `-webkit-border-radius` also follow the same rule, for instance: +Browser-prefixed properties like `-moz-border-radius`, `-webkit-border-radius` also follow the same rule: a dash means upper case. + +For instance: ```js button.style.MozBorderRadius = '5px'; button.style.WebkitBorderRadius = '5px'; ``` - -That is: a dash `"-"` becomes an uppercase. ```` ## Resetting the style property @@ -119,13 +120,21 @@ For instance, to hide an element, we can set `elem.style.display = "none"`. Then later we may want to remove the `style.display` as if it were not set. Instead of `delete elem.style.display` we should assign an empty string to it: `elem.style.display = ""`. ```js run -// if we run this code, the <body> "blinks" +// if we run this code, the <body> will blink document.body.style.display = "none"; // hide setTimeout(() => document.body.style.display = "", 1000); // back to normal ``` -If we set `display` to an empty string, then the browser applies CSS classes and its built-in styles normally, as if there were no such `display` property at all. +If we set `style.display` to an empty string, then the browser applies CSS classes and its built-in styles normally, as if there were no such `style.display` property at all. + +Also there is a special method for that, `elem.style.removeProperty('style property')`. So, We can remove a property like this: + +```js run +document.body.style.background = 'red'; //set background to red + +setTimeout(() => document.body.style.removeProperty('background'), 1000); // remove background after 1 second +``` ````smart header="Full rewrite with `style.cssText`" Normally, we use `style.*` to assign individual style properties. We can't set the full style like `div.style="color: red; width: 100px"`, because `div.style` is an object, and it's read-only. @@ -147,14 +156,14 @@ To set the full style as a string, there's a special property `style.cssText`: </script> ``` -We rarely use it, because such assignment removes all existing styles: it does not add, but replaces them. May occasionally delete something needed. But still can be done for new elements when we know we won't delete something important. +This property is rarely used, because such assignment removes all existing styles: it does not add, but replaces them. May occasionally delete something needed. But we can safely use it for new elements, when we know we won't delete an existing style. The same can be accomplished by setting an attribute: `div.setAttribute('style', 'color: red...')`. ```` ## Mind the units -CSS units must be provided in style values. +Don't forget to add CSS units to values. For instance, we should not set `elem.style.top` to `10`, but rather to `10px`. Otherwise it wouldn't work: @@ -177,11 +186,11 @@ For instance, we should not set `elem.style.top` to `10`, but rather to `10px`. </body> ``` -Please note how the browser "unpacks" the property `style.margin` in the last lines and infers `style.marginLeft` and `style.marginTop` (and other partial margins) from it. +Please note: the browser "unpacks" the property `style.margin` in the last lines and infers `style.marginLeft` and `style.marginTop` from it. ## Computed styles: getComputedStyle -Modifying a style is easy. But how to *read* it? +So, modifying a style is easy. But how to *read* it? For instance, we want to know the size, margins, the color of an element. How to do it? @@ -207,14 +216,14 @@ For instance, here `style` doesn't see the margin: </body> ``` -...But what if we need, say, to increase the margin by 20px? We would want the current value of it. +...But what if we need, say, to increase the margin by `20px`? We would want the current value of it. There's another method for that: `getComputedStyle`. The syntax is: ```js -getComputedStyle(element[, pseudo]) +getComputedStyle(element, [pseudo]) ``` element @@ -223,7 +232,7 @@ element pseudo : A pseudo-element if required, for instance `::before`. An empty string or no argument means the element itself. -The result is an object with style properties, like `elem.style`, but now with respect to all CSS classes. +The result is an object with styles, like `elem.style`, but now with respect to all CSS classes. For instance: @@ -248,40 +257,26 @@ For instance: ```smart header="Computed and resolved values" There are two concepts in [CSS](https://drafts.csswg.org/cssom/#resolved-values): -1. A *computed* style value is the value after all CSS rules and CSS inheritance is applied, as the result of the CSS cascade. It can look like `height:1em` or `font-size:125%`. +1. A *computed* style value is the value after all CSS rules and CSS inheritance is applied, as the result of the CSS cascade. It can look like `height:1em` or `font-size:125%`. 2. A *resolved* style value is the one finally applied to the element. Values like `1em` or `125%` are relative. The browser takes the computed value and makes all units fixed and absolute, for instance: `height:20px` or `font-size:16px`. For geometry properties resolved values may have a floating point, like `width:50.5px`. A long time ago `getComputedStyle` was created to get computed values, but it turned out that resolved values are much more convenient, and the standard changed. -So nowadays `getComputedStyle` actually returns the resolved value of the property. +So nowadays `getComputedStyle` actually returns the resolved value of the property, usually in `px` for geometry. ``` ````warn header="`getComputedStyle` requires the full property name" We should always ask for the exact property that we want, like `paddingLeft` or `marginTop` or `borderTopWidth`. Otherwise the correct result is not guaranteed. For instance, if there are properties `paddingLeft/paddingTop`, then what should we get for `getComputedStyle(elem).padding`? Nothing, or maybe a "generated" value from known paddings? There's no standard rule here. - -There are other inconsistencies. As an example, some browsers (Chrome) show `10px` in the document below, and some of them (Firefox) -- do not: - -```html run -<style> - body { - margin: 10px; - } -</style> -<script> - let style = getComputedStyle(document.body); - alert(style.margin); // empty string in Firefox -</script> -``` ```` -```smart header="\"Visited\" links styles are hidden!" +```smart header="Styles applied to `:visited` links are hidden!" Visited links may be colored using `:visited` CSS pseudoclass. But `getComputedStyle` does not give access to that color, because otherwise an arbitrary page could find out whether the user visited a link by creating it on the page and checking the styles. -JavaScript may not see the styles applied by `:visited`. And also, there's a limitation in CSS that forbids to apply geometry-changing styles in `:visited`. That's to guarantee that there's no sideway for an evil page to test if a link was visited and hence to break the privacy. +JavaScript may not see the styles applied by `:visited`. And also, there's a limitation in CSS that forbids applying geometry-changing styles in `:visited`. That's to guarantee that there's no side way for an evil page to test if a link was visited and hence to break the privacy. ``` ## Summary @@ -299,4 +294,4 @@ To change the styles: To read the resolved styles (with respect to all classes, after all CSS is applied and final values are calculated): -- The `getComputedStyle(elem[, pseudo])` returns the style-like object with them. Read-only. +- The `getComputedStyle(elem, [pseudo])` returns the style-like object with them. Read-only. diff --git a/2-ui/1-document/09-size-and-scroll/1-get-scroll-height-bottom/task.md b/2-ui/1-document/09-size-and-scroll/1-get-scroll-height-bottom/task.md index b0f1f2214..796039c2a 100644 --- a/2-ui/1-document/09-size-and-scroll/1-get-scroll-height-bottom/task.md +++ b/2-ui/1-document/09-size-and-scroll/1-get-scroll-height-bottom/task.md @@ -4,7 +4,7 @@ importance: 5 # What's the scroll from the bottom? -The `elem.scrollTop` property is the size of the scrolled out part from the top. How to get "`scrollBottom`" -- the size from the bottom? +The `elem.scrollTop` property is the size of the scrolled out part from the top. How to get the size of the bottom scroll (let's call it `scrollBottom`)? Write the code that works for an arbitrary `elem`. diff --git a/2-ui/1-document/09-size-and-scroll/2-scrollbar-width/task.md b/2-ui/1-document/09-size-and-scroll/2-scrollbar-width/task.md index 67787894a..b39004cbf 100644 --- a/2-ui/1-document/09-size-and-scroll/2-scrollbar-width/task.md +++ b/2-ui/1-document/09-size-and-scroll/2-scrollbar-width/task.md @@ -6,6 +6,6 @@ importance: 3 Write the code that returns the width of a standard scrollbar. -For Windows it usually varies between `12px` and `20px`. If the browser doesn't reserves any space for it, then it may be `0px`. +For Windows it usually varies between `12px` and `20px`. If the browser doesn't reserve any space for it (the scrollbar is half-translucent over the text, also happens), then it may be `0px`. P.S. The code should work for any HTML document, do not depend on its content. diff --git a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/ball-half/index.html b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/ball-half/index.html index ca9c4d579..8f855ecfa 100755 --- a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/ball-half/index.html +++ b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/ball-half/index.html @@ -9,7 +9,7 @@ background-color: #00FF00; position: relative; } - + #ball { position: absolute; } @@ -20,7 +20,7 @@ <div id="field"> - <img src="https://js.cx/clipart/ball.svg" id="ball"> . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . + <img src="https://js.cx/clipart/ball.svg" height="40" width="40" id="ball"> . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . </div> @@ -38,4 +38,4 @@ </body> -</html> \ No newline at end of file +</html> diff --git a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/field.svg b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/field.svg index 4ae90b1c7..448150682 100644 --- a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/field.svg +++ b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/field.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.md b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.md index 1dce9ffaa..afa1d8f50 100644 --- a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.md +++ b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.md @@ -24,19 +24,22 @@ ball.style.left = Math.round(field.clientWidth / 2 - ball.offsetWidth / 2) + 'px ball.style.top = Math.round(field.clientHeight / 2 - ball.offsetHeight / 2) + 'px'; ``` -**Attention: the pitfall!** +Now the ball is finally centered. + +````warn header="Attention: the pitfall!" The code won't work reliably while `<img>` has no width/height: ```html <img src="ball.png" id="ball"> ``` +```` When the browser does not know the width/height of an image (from tag attributes or CSS), then it assumes them to equal `0` until the image finishes loading. -In real life after the first load browser usually caches the image, and on next loads it will have the size immediately. +So the value of `ball.offsetWidth` will be `0` until the image loads. That leads to wrong coordinates in the code above. -But on the first load the value of `ball.offsetWidth` is `0`. That leads to wrong coordinates. +After the first load, the browser usually caches the image, and on reloads it will have the size immediately. But on the first load the value of `ball.offsetWidth` is `0`. We should fix that by adding `width/height` to `<img>`: diff --git a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.view/index.html b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.view/index.html index 9ebe6001e..9f21e5421 100755 --- a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.view/index.html +++ b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/solution.view/index.html @@ -26,7 +26,7 @@ <script> - // ball.offsetWidth=0 before image loaded! + // ball.offsetWidth=0 before image loaded! // to fix: set width ball.style.left = Math.round(field.clientWidth / 2 - ball.offsetWidth / 2) + 'px' ball.style.top = Math.round(field.clientHeight / 2 - ball.offsetHeight / 2) + 'px' diff --git a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/task.md b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/task.md index 57746d921..f56e0858b 100644 --- a/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/task.md +++ b/2-ui/1-document/09-size-and-scroll/4-put-ball-in-center/task.md @@ -10,7 +10,7 @@ Here's how the source document looks: What are coordinates of the field center? -Calculate them and use to place the ball into the center of the field: +Calculate them and use to place the ball into the center of the green field: [iframe src="solution" height=180] diff --git a/2-ui/1-document/09-size-and-scroll/article.md b/2-ui/1-document/09-size-and-scroll/article.md index 88d671604..872e70dda 100644 --- a/2-ui/1-document/09-size-and-scroll/article.md +++ b/2-ui/1-document/09-size-and-scroll/article.md @@ -2,8 +2,7 @@ There are many JavaScript properties that allow us to read information about element width, height and other geometry features. -We often need them when moving or positioning elements in JavaScript, to correctly calculate coordinates. - +We often need them when moving or positioning elements in JavaScript. ## Sample element @@ -18,8 +17,8 @@ As a sample element to demonstrate properties we'll use the one given below: width: 300px; height: 200px; border: 25px solid #E8C48F; - padding: 20px; - overflow: auto; + padding: 20px; + overflow: auto; } </style> ``` @@ -33,38 +32,42 @@ The element looks like this: You can [open the document in the sandbox](sandbox:metric). ```smart header="Mind the scrollbar" -The picture above demonstrates the most complex case when the element has a scrollbar. Some browsers (not all) reserve the space for it by taking it from the content. +The picture above demonstrates the most complex case when the element has a scrollbar. Some browsers (not all) reserve the space for it by taking it from the content (labeled as "content width" above). -So, without scrollbar the content width would be `300px`, but if the scrollbar is `16px` wide (the width may vary between devices and browsers) then only `300 - 16 = 284px` remains, and we should take it into account. That's why examples from this chapter assume that there's a scrollbar. If there's no scrollbar, then things are just a bit simpler. +So, without scrollbar the content width would be `300px`, but if the scrollbar is `16px` wide (the width may vary between devices and browsers) then only `300 - 16 = 284px` remains, and we should take it into account. That's why examples from this chapter assume that there's a scrollbar. Without it, some calculations are simpler. ``` -```smart header="The `padding-bottom` may be filled with text" -Usually paddings are shown empty on illustrations, but if there's a lot of text in the element and it overflows, then browsers show the "overflowing" text at `padding-bottom`, so you can see that in examples. But the padding is still there, unless specified otherwise. +```smart header="The `padding-bottom` area may be filled with text" +Usually paddings are shown empty on our illustrations, but if there's a lot of text in the element and it overflows, then browsers show the "overflowing" text at `padding-bottom`, that's normal. ``` ## Geometry -Element properties that provide width, height and other geometry are always numbers. They are assumed to be in pixels. - -Here's the overall picture: +Here's the overall picture with geometry properties:  -They are many properties, it's difficult to fit them all in the single picture, but their values are simple and easy to understand. +<<<<<<< HEAD + +======= +Values of these properties are technically numbers, but these numbers are "of pixels", so these are pixel measurements. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -Let's start exploring them from the outside of the element. +Let's start exploring the properties starting from the outside of the element. ## offsetParent, offsetLeft/Top These properties are rarely needed, but still they are the "most outer" geometry properties, so we'll start with them. -The `offsetParent` is the nearest ancestor that is: +The `offsetParent` is the nearest ancestor that the browser uses for calculating coordinates during rendering. -1. CSS-positioned (`position` is `absolute`, `relative`, `fixed` or `sticky`), -2. or `<td>`, `<th>`, `<table>`, -2. or `<body>`. +That's the nearest ancestor that is one of the following: -In most practical cases we can use `offsetParent` to get the nearest CSS-positioned ancestor. And `offsetLeft/offsetTop` provide x/y coordinates relative to its upper-left corner. +1. CSS-positioned (`position` is `absolute`, `relative`, `fixed` or `sticky`), or +2. `<td>`, `<th>`, or `<table>`, or +3. `<body>`. + +Properties `offsetLeft/offsetTop` provide x/y coordinates relative to `offsetParent` upper-left corner. In the example below the inner `<div>` has `<main>` as `offsetParent` and `offsetLeft/offsetTop` shifts from its upper-left corner (`180`): @@ -82,7 +85,10 @@ In the example below the inner `<div>` has `<main>` as `offsetParent` and `offse ```  +<<<<<<< HEAD +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b There are several occasions when `offsetParent` is `null`: @@ -103,12 +109,12 @@ For our sample element: - `offsetWidth = 390` -- the outer width, can be calculated as inner CSS-width (`300px`) plus paddings (`2 * 20px`) and borders (`2 * 25px`). - `offsetHeight = 290` -- the outer height. -````smart header="Geometry properties for not shown elements are zero/null" -Geometry properties are calculated only for shown elements. +````smart header="Geometry properties are zero/null for elements that are not displayed" +Geometry properties are calculated only for displayed elements. -If an element (or any of its ancestors) has `display:none` or is not in the document, then all geometry properties are zero or `null` depending on what it is. +If an element (or any of its ancestors) has `display:none` or is not in the document, then all geometry properties are zero (or `null` for `offsetParent`). -For example, `offsetParent` is `null`, and `offsetWidth`, `offsetHeight` are `0`. +For example, `offsetParent` is `null`, and `offsetWidth`, `offsetHeight` are `0` when we created an element, but haven't inserted it into the document yet, or it (or its ancestor) has `display:none`. We can use this to check if an element is hidden, like this: @@ -118,7 +124,7 @@ function isHidden(elem) { } ``` -Please note that such `isHidden` returns `true` for elements that are on-screen, but have zero sizes (like an empty `<div>`). +Please note that such `isHidden` returns `true` for elements that are on-screen, but have zero sizes. ```` ## clientTop/Left @@ -134,14 +140,19 @@ In our example:  -...But to be precise -- they are not borders, but relative coordinates of the inner side from the outer side. +...But to be precise -- these properties are not border width/height, but rather relative coordinates of the inner side from the outer side. What's the difference? It becomes visible when the document is right-to-left (the operating system is in Arabic or Hebrew languages). The scrollbar is then not on the right, but on the left, and then `clientLeft` also includes the scrollbar width. -In that case, `clientLeft` would be not `25`, but with the scrollbar width `25 + 16 = 41`: +In that case, `clientLeft` would be not `25`, but with the scrollbar width `25 + 16 = 41`. + +<<<<<<< HEAD +======= +Here's the example in hebrew: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  ## clientWidth/Height @@ -152,7 +163,9 @@ They include the content width together with paddings, but without the scrollbar  -On the picture above let's first consider `clientHeight`: it's easier to evaluate. There's no horizontal scrollbar, so it's exactly the sum of what's inside the borders: CSS-height `200px` plus top and bottom paddings (`2 * 20px`) total `240px`. +On the picture above let's first consider `clientHeight`. + +There's no horizontal scrollbar, so it's exactly the sum of what's inside the borders: CSS-height `200px` plus top and bottom paddings (`2 * 20px`) total `240px`. Now `clientWidth` -- here the content width is not `300px`, but `284px`, because `16px` are occupied by the scrollbar. So the sum is `284px` plus left and right paddings, total `324px`. @@ -164,8 +177,7 @@ So when there's no padding we can use `clientWidth/clientHeight` to get the cont ## scrollWidth/Height -- Properties `clientWidth/clientHeight` only account for the visible part of the element. -- Properties `scrollWidth/scrollHeight` also include the scrolled out (hidden) parts: +These properties are like `clientWidth/clientHeight`, but they also include the scrolled out (hidden) parts:  @@ -210,16 +222,16 @@ If you click the element below, the code `elem.scrollTop += 10` executes. That m <div onclick="this.scrollTop+=10" style="cursor:pointer;border:1px solid black;width:100px;height:80px;overflow:auto">Click<br>Me<br>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9</div> ``` -Setting `scrollTop` to `0` or `Infinity` will make the element scroll to the very top/bottom respectively. +Setting `scrollTop` to `0` or a big value, such as `1e9` will make the element scroll to the very top/bottom respectively. ```` ## Don't take width/height from CSS -We've just covered geometry properties of DOM elements. They are normally used to get widths, heights and calculate distances. +We've just covered geometry properties of DOM elements, that can be used to get widths, heights and calculate distances. But as we know from the chapter <info:styles-and-classes>, we can read CSS-height and width using `getComputedStyle`. -So why not to read the width of an element like this? +So why not to read the width of an element with `getComputedStyle`, like this? ```js run let elem = document.body; @@ -229,7 +241,7 @@ alert( getComputedStyle(elem).width ); // show CSS width for elem Why should we use geometry properties instead? There are two reasons: -1. First, CSS width/height depend on another property: `box-sizing` that defines "what is" CSS width and height. A change in `box-sizing` for CSS purposes may break such JavaScript. +1. First, CSS `width/height` depend on another property: `box-sizing` that defines "what is" CSS width and height. A change in `box-sizing` for CSS purposes may break such JavaScript. 2. Second, CSS `width/height` may be `auto`, for instance for an inline element: ```html run @@ -242,9 +254,9 @@ Why should we use geometry properties instead? There are two reasons: </script> ``` - From the CSS standpoint, `width:auto` is perfectly normal, but in JavaScript we need an exact size in `px` that we can use in calculations. So here CSS width is useless at all. + From the CSS standpoint, `width:auto` is perfectly normal, but in JavaScript we need an exact size in `px` that we can use in calculations. So here CSS width is useless. -And there's one more reason: a scrollbar. Sometimes the code that works fine without a scrollbar starts to bug with it, because a scrollbar takes the space from the content in some browsers. So the real width available for the content is *less* than CSS width. And `clientWidth/clientHeight` take that into account. +And there's one more reason: a scrollbar. Sometimes the code that works fine without a scrollbar becomes buggy with it, because a scrollbar takes the space from the content in some browsers. So the real width available for the content is *less* than CSS width. And `clientWidth/clientHeight` take that into account. ...But with `getComputedStyle(elem).width` the situation is different. Some browsers (e.g. Chrome) return the real inner width, minus the scrollbar, and some of them (e.g. Firefox) -- CSS width (ignore the scrollbar). Such cross-browser differences is the reason not to use `getComputedStyle`, but rather rely on geometry properties. @@ -267,9 +279,9 @@ Elements have the following geometry properties: - `offsetParent` -- is the nearest positioned ancestor or `td`, `th`, `table`, `body`. - `offsetLeft/offsetTop` -- coordinates relative to the upper-left edge of `offsetParent`. - `offsetWidth/offsetHeight` -- "outer" width/height of an element including borders. -- `clientLeft/clientTop` -- the distance from the upper-left outer corner to its upper-left inner corner. For left-to-right OS they are always the widths of left/top borders. For right-to-left OS the vertical scrollbar is on the left so `clientLeft` includes its width too. +- `clientLeft/clientTop` -- the distances from the upper-left outer corner to the upper-left inner (content + padding) corner. For left-to-right OS they are always the widths of left/top borders. For right-to-left OS the vertical scrollbar is on the left so `clientLeft` includes its width too. - `clientWidth/clientHeight` -- the width/height of the content including paddings, but without the scrollbar. -- `scrollWidth/scrollHeight` -- the width/height of the content including the scrolled out parts. Also includes paddings, but not the scrollbar. -- `scrollLeft/scrollTop` -- width/height of the scrolled out part of the element, starting from its upper-left corner. +- `scrollWidth/scrollHeight` -- the width/height of the content, just like `clientWidth/clientHeight`, but also include scrolled-out, invisible part of the element. +- `scrollLeft/scrollTop` -- width/height of the scrolled out upper part of the element, starting from its upper-left corner. -All properties are read-only except `scrollLeft/scrollTop`. They make the browser scroll the element if changed. +All properties are read-only except `scrollLeft/scrollTop` that make the browser scroll the element if changed. diff --git a/2-ui/1-document/09-size-and-scroll/metric-all.svg b/2-ui/1-document/09-size-and-scroll/metric-all.svg index a5dadb47f..ad20c645a 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-all.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-all.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="670" height="602" viewBox="0 0 670 602"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-all.svg"><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="160" y="94">Introduction</tspan> <tspan x="160" y="122" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="160" y="141" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="160" y="160" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="160" y="179" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="160" y="198" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="160" y="217" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="160" y="236" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="160" y="255" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="160" y="274" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="160" y="293" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="160" y="312" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="160" y="331" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="160" y="350" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="160" y="369" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="160" y="388" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="160" y="407" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="160" y="426" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="160" y="445" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="160" y="464" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="160" y="483" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="160" y="502" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="160" y="521" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="160" y="540" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M491 162v290H117V162h374zm-25 25H142v240h324V187z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M141 62h326v500H141z"/><text id="scrollHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 592 310)"><tspan x="541.6" y="314.5">scrollHeight</tspan></text><text id="offsetHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 552 310)"><tspan x="501.6" y="314.5">offsetHeight</tspan></text><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 618 125)"><tspan x="580.2" y="129.5">scrollTop</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 62H640"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 163h92.14"/><path id="Line-29" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 451h92.14"/><path id="Line-33" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 189H640"/><path id="Line-32" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 427h72.14"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 561h148.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M605 64.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><path id="Line-30" fill="#C06334" fill-rule="nonzero" d="M565 164.5l7 14h-6v255h6l-7 14-7-14h6v-255h-6l7-14z"/><text id="clientHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 510 304)"><tspan x="459.6" y="308.5">clientHeight</tspan></text><path id="Line-34" fill="#C06334" fill-rule="nonzero" d="M523 191.5l7 14h-6v206h6l-7 14-7-14h6v-206h-6l7-14z"/><text id="offsetTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 104 83)"><tspan x="66.2" y="87.5">offsetTop</tspan></text><text id="clientLeft" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 130.5 237)"><tspan x="88.5" y="241.5">clientLeft</tspan></text><path id="Line-36" fill="#C06334" fill-rule="nonzero" d="M117 4.5l7 14h-6v128h6l-7 14-7-14h6v-128h-6l7-14z"/><path id="Line-31" fill="#C06334" fill-rule="nonzero" d="M631 64.5l7 14h-6v96.499l6 .001-7 14-7-14 6-.001V78.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M154 73h312v89H154z"/><path id="Rectangle-15" fill="#FFF" d="M154 451h312v93H154z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M431 479.09l14 7-14 7-.001-6h-271.36l.001 6-14-7 14-7-.001 6h271.36l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M445.64 510v-84"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M141.64 510v-84"/><text id="clientWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="261.3" y="478">clientWidth</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M100 156.09l14 7-14 7v-6H18.639l.001 6-14-7 14-7-.001 6H100v-6z"/><text id="clientTop" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="147.7" y="178">clientTop</tspan></text><text id="offsetLeft" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="17.5" y="154">offsetLeft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M475 522.09l14 7-14 7-.001-6h-340.36l.001 6-14-7 14-7-.001 6h340.36l.001-6z"/><path id="Line-45" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M490.64 551V447"/><path id="Line-44" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M116.64 551V447"/><text id="offsetWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="258.3" y="516">offsetWidth</tspan></text><path id="Rectangle-233" stroke="#AF6E24" stroke-width="2" d="M1 1h668v600H1z"/><g id="Group" transform="translate(450 187)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><g id="Group" transform="translate(115.676 162.5)"><g id="Line-4-+-Line-5" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="translate(22.324 18.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><g id="Line-4-+-Line-6" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="matrix(1 0 0 -1 22.324 6.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><path id="Line-49" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.824 25.5h-25"/><path id="Line-50" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.324 25V0"/><path id="Line-4" fill="#166388" fill-rule="nonzero" d="M19.328 21.676l.91.414 5.5 2.5.91.413-.589 1.296.09.198-.226.1-.102.227-.198-.09-.385.176-5.5 2.5-.91.414-.828-1.82.91-.414 4.297-1.954-3.797-1.726-.91-.413.828-1.821zM7.32 21.676l.828 1.82-.91.414-3.798 1.726 4.298 1.954.91.413-.827 1.821-.91-.414-5.5-2.5-.387-.176-.196.09-.103-.225-.225-.102.089-.198L0 25.003l.91-.413 5.5-2.5.91-.414z"/></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="670" height="602" viewBox="0 0 670 602"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-all.svg"><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="160" y="94">Introduction</tspan> <tspan x="160" y="122" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="160" y="141" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="160" y="160" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="160" y="179" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="160" y="198" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="160" y="217" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="160" y="236" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="160" y="255" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="160" y="274" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="160" y="293" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="160" y="312" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="160" y="331" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="160" y="350" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="160" y="369" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="160" y="388" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="160" y="407" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="160" y="426" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="160" y="445" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="160" y="464" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="160" y="483" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="160" y="502" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="160" y="521" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="160" y="540" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M491 162v290H117V162h374zm-25 25H142v240h324V187z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M141 62h326v500H141z"/><text id="scrollHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 592 310)"><tspan x="541.6" y="314.5">scrollHeight</tspan></text><text id="offsetHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 552 310)"><tspan x="501.6" y="314.5">offsetHeight</tspan></text><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 618 125)"><tspan x="580.2" y="129.5">scrollTop</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 62H640"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 163h92.14"/><path id="Line-29" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 451h92.14"/><path id="Line-33" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 189H640"/><path id="Line-32" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 427h72.14"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 561h148.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M605 64.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><path id="Line-30" fill="#C06334" fill-rule="nonzero" d="M565 164.5l7 14h-6v255h6l-7 14-7-14h6v-255h-6l7-14z"/><text id="clientHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 510 304)"><tspan x="459.6" y="308.5">clientHeight</tspan></text><path id="Line-34" fill="#C06334" fill-rule="nonzero" d="M523 191.5l7 14h-6v206h6l-7 14-7-14h6v-206h-6l7-14z"/><text id="offsetTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 104 83)"><tspan x="66.2" y="87.5">offsetTop</tspan></text><text id="clientLeft" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 130.5 237)"><tspan x="88.5" y="241.5">clientLeft</tspan></text><path id="Line-36" fill="#C06334" fill-rule="nonzero" d="M117 4.5l7 14h-6v128h6l-7 14-7-14h6v-128h-6l7-14z"/><path id="Line-31" fill="#C06334" fill-rule="nonzero" d="M631 64.5l7 14h-6v96.499l6 .001-7 14-7-14 6-.001V78.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M154 73h312v89H154z"/><path id="Rectangle-15" fill="#FFF" d="M154 451h312v93H154z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M431 479.09l14 7-14 7-.001-6h-271.36l.001 6-14-7 14-7-.001 6h271.36l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M445.64 510v-84"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M141.64 510v-84"/><text id="clientWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="261.3" y="478">clientWidth</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M100 156.09l14 7-14 7v-6H18.639l.001 6-14-7 14-7-.001 6H100v-6z"/><text id="clientTop" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="147.7" y="178">clientTop</tspan></text><text id="offsetLeft" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="17.5" y="154">offsetLeft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M475 522.09l14 7-14 7-.001-6h-340.36l.001 6-14-7 14-7-.001 6h340.36l.001-6z"/><path id="Line-45" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M490.64 551V447"/><path id="Line-44" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M116.64 551V447"/><text id="offsetWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="258.3" y="516">offsetWidth</tspan></text><path id="Rectangle-233" stroke="#AF6E24" stroke-width="2" d="M1 1h668v600H1z"/><g id="Group" transform="translate(450 187)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><g id="Group" transform="translate(115.676 162.5)"><g id="Line-4-+-Line-5" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="translate(22.324 18.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><g id="Line-4-+-Line-6" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="matrix(1 0 0 -1 22.324 6.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><path id="Line-49" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.824 25.5h-25"/><path id="Line-50" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.324 25V0"/><path id="Line-4" fill="#166388" fill-rule="nonzero" d="M19.328 21.676l.91.414 5.5 2.5.91.413-.589 1.296.09.198-.226.1-.102.227-.198-.09-.385.176-5.5 2.5-.91.414-.828-1.82.91-.414 4.297-1.954-3.797-1.726-.91-.413.828-1.821zM7.32 21.676l.828 1.82-.91.414-3.798 1.726 4.298 1.954.91.413-.827 1.821-.91-.414-5.5-2.5-.387-.176-.196.09-.103-.225-.225-.102.089-.198L0 25.003l.91-.413 5.5-2.5.91-.414z"/></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="670" height="602" viewBox="0 0 670 602"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-all.svg"><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="160" y="94">Introduction</tspan> <tspan x="160" y="122" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="160" y="141" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="160" y="160" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="160" y="179" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="160" y="198" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="160" y="217" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="160" y="236" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="160" y="255" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="160" y="274" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="160" y="293" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="160" y="312" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="160" y="331" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="160" y="350" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="160" y="369" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="160" y="388" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="160" y="407" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="160" y="426" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="160" y="445" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="160" y="464" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="160" y="483" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="160" y="502" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="160" y="521" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="160" y="540" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M491 162v290H117V162h374zm-25 25H142v240h324V187z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M141 62h326v500H141z"/><text id="scrollHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 592 310)"><tspan x="541.6" y="314.5">scrollHeight</tspan></text><text id="offsetHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 552 310)"><tspan x="501.6" y="314.5">offsetHeight</tspan></text><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 618 125)"><tspan x="580.2" y="129.5">scrollTop</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 62H640"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 163h92.14"/><path id="Line-29" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M492.5 451h92.14"/><path id="Line-33" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 189H640"/><path id="Line-32" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M467.5 427h72.14"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M466.5 561h148.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M605 64.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><path id="Line-30" fill="#C06334" fill-rule="nonzero" d="M565 164.5l7 14h-6v255h6l-7 14-7-14h6v-255h-6l7-14z"/><text id="clientHeight" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 510 304)"><tspan x="459.6" y="308.5">clientHeight</tspan></text><path id="Line-34" fill="#C06334" fill-rule="nonzero" d="M523 191.5l7 14h-6v206h6l-7 14-7-14h6v-206h-6l7-14z"/><text id="offsetTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 104 83)"><tspan x="66.2" y="87.5">offsetTop</tspan></text><text id="clientLeft" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 130.5 237)"><tspan x="88.5" y="241.5">clientLeft</tspan></text><path id="Line-36" fill="#C06334" fill-rule="nonzero" d="M117 4.5l7 14h-6v128h6l-7 14-7-14h6v-128h-6l7-14z"/><path id="Line-31" fill="#C06334" fill-rule="nonzero" d="M631 64.5l7 14h-6v96.499l6 .001-7 14-7-14 6-.001V78.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M154 73h312v89H154z"/><path id="Rectangle-15" fill="#FFF" d="M154 451h312v93H154z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M431 479.09l14 7-14 7-.001-6h-271.36l.001 6-14-7 14-7-.001 6h271.36l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M445.64 510v-84"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M141.64 510v-84"/><text id="clientWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="261.3" y="478">clientWidth</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M100 156.09l14 7-14 7v-6H18.639l.001 6-14-7 14-7-.001 6H100v-6z"/><text id="clientTop" fill="#166388" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="147.7" y="178">clientTop</tspan></text><text id="offsetLeft" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="17.5" y="154">offsetLeft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M475 522.09l14 7-14 7-.001-6h-340.36l.001 6-14-7 14-7-.001 6h340.36l.001-6z"/><path id="Line-45" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M490.64 551V447"/><path id="Line-44" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M116.64 551V447"/><text id="offsetWidth" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="258.3" y="516">offsetWidth</tspan></text><path id="Rectangle-233" stroke="#AF6E24" stroke-width="2" d="M1 1h668v600H1z"/><g id="Scrollbar" transform="translate(450 187)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><g id="Group" transform="translate(115.676 162.5)"><g id="Line-4-+-Line-5" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="translate(22.324 18.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><g id="Line-4-+-Line-6" stroke="#166388" stroke-linecap="square" stroke-width="2" transform="matrix(1 0 0 -1 22.324 6.5)"><path id="Line-4" d="M2.5.5L0 6" transform="matrix(1 0 0 -1 0 6)"/><path id="Line-5" d="M5.5.5L3 6" transform="rotate(180 4.5 3)"/></g><path id="Line-49" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.824 25.5h-25"/><path id="Line-50" fill="#166388" stroke="#166388" stroke-linecap="round" stroke-linejoin="bevel" stroke-width="2" d="M25.324 25V0"/><path id="Line-4" fill="#166388" fill-rule="nonzero" d="M19.328 21.676l.91.414 5.5 2.5.91.413-.589 1.296.09.198-.226.1-.102.227-.198-.09-.385.176-5.5 2.5-.91.414-.828-1.82.91-.414 4.297-1.954-3.797-1.726-.91-.413.828-1.821zM7.32 21.676l.828 1.82-.91.414-3.798 1.726 4.298 1.954.91.413-.827 1.821-.91-.414-5.5-2.5-.387-.176-.196.09-.103-.225-.225-.102.089-.198L0 25.003l.91-.413 5.5-2.5.91-.414z"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-client-left-top-rtl.svg b/2-ui/1-document/09-size-and-scroll/metric-client-left-top-rtl.svg index dd9e17cf8..e62609f68 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-client-left-top-rtl.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-client-left-top-rtl.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="359" height="316" viewBox="0 0 359 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top-rtl.svg"><path fill="#FFF" d="M0 0h359v316H0z"/><path id="היא-שפת-תסריט-מפורשת" fill="#643B0C" d="M336.484 135v-10.031h4.047V135h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047zm-20.39-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM302.063 135v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V135h-4.329zm-31.954 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.718 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L262.671 135h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.516 135h-8.313zm-33.547 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L199.969 135h-8.313zm-18.265 0l-1.766-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.876.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.343 5.156c-2.26 1.313-5.22 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.907-3.25 2.907-5.531 0-.938-.328-1.57-.985-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.157 9.453zM162.297 135v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.531-4.625h5.047c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.148V135h-4.047zm-14.047-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.375 135l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm222.344 38l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.757l.531 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L348.953 173h-4.234zm-19.344 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.874 173h-14.5zm-7.14 0v-9.594c0-1.927-.32-3.252-.962-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V173h-4.046zm-30.516 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zM250 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L258.313 173H250zm-31.422 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L222.813 173h-4.234zm-19.25 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L215.53 173h-16.203zm-7.516 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-23.015 0l-1.766-14.078h-2.515l1.547-4.625h11.078c2.093 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.105 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zM146.531 173l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.453-4.625c2.334-.177 4.089-.687 5.266-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zM119.531 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 173h-8.313zm288.766 38v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.665l-.703-.671c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V211h-4.328zm-8.438 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-23.718 0l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L392.344 211H376.14zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.594 0h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.546-4.625h10.844c1.99 0 3.39.088 4.203.266.813.177 1.495.541 2.047 1.093.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM312.594 211l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-6.516-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM277.766 211v-14.078h-2.485l1.547-4.625h10.485c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L293.578 211h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zM243.86 211v-10.031h4.047V211h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V211h-4.047zm-37.594 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L224.094 211h-4.235zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L192.969 211h-8.313zm-19.703 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.079-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.304-1.927.304-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.546 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V211h-4.329zm-22.14 0l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L147.047 211h-4.234zm-23.282 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 211h-8.313zm318.844 38.39l1.594-4.718c2.041-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.86-.39l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.28 2.312c2.72 0 4.891-.453 6.517-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-6.546-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zm-24.016 6.657l1.594-4.72c2.042-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-8.89-.391v-9.594c0-1.927-.32-3.252-.961-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V249h-4.046zm-23.72 0l1.548-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L369.547 249h-16.203zm-29.859 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.608-4.32-.407-.463-1.683-.695-3.829-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L339.688 249h-16.204zm-20.046 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-24.266 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.329l1.547-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V249h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L287.484 249h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-14.047-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 249v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 249l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.453v7.188L202.328 249h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L149.86 249h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm258.422 38v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.156V287h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V287h-4.047zm-29.25 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-20.594 0v-10.031h4.047V287h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V287h-4.047zm-48.61 2.781l1.563-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.265 14.563c1.636-1.24 2.774-2.78 3.415-4.625.64-1.843.96-4.484.96-7.921v-1.641h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zm-14.796-2.39l1.594-4.72c2.041-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.642-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-15.938-6.657h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM236.656 287v-14.078h-9.531l1.547-4.625h16.094l-1.547 4.625h-2.516V287h-4.047zm-12.86-6.266h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM196.547 287v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L182.796 287h-14.5zm-8.75 0v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L145.796 287h-14.5zm-7.155 4.625v-14.281c0-2.032-.36-3.344-1.079-3.938-1.041-.875-2.328-1.552-3.859-2.031l1.86-3.453c1.875.562 3.53 1.37 4.968 2.422.834.614 1.401 1.364 1.703 2.25.302.885.454 2.255.454 4.11v14.921h-4.047zM426.797 325v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.156V325h-4.046zm-36.422 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V325h-4.328zm-4.938-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM376.86 325l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.454v7.187L386.438 325h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm-9.937 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V325h-4.046zm-24.625 0l1.562-4.625h5.516v-6.078c0-1.646-.354-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.297 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L333.969 325h-9.578zm-17.907 0l-2.234-17.688 3.922-1.39 1.797 14.453c2.666 0 4.794-.615 6.383-1.844 1.588-1.229 2.382-2.87 2.382-4.922 0-2.24-1.208-3.359-3.625-3.359-.645 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.917 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437-1.395 2-3.291 3.568-5.687 4.704-2.01.958-4.849 1.437-8.516 1.437zm-32.359 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.197.797.296 2.18.296 4.149v6.562h2.5L290.33 325h-16.204zm-19.219 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.094-1.578a28.973 28.973 0 004.937 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-8.687 0v-14.078h-9.531l1.546-4.625h16.094l-1.547 4.625h-2.515V325h-4.047zm-40.422 0v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V325h-4.047zm-32.813.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.125 4.235v-14.61h4.047v14.61h-4.047zm-2.516-18.703l1.547-4.625h10.844c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM155.625 325v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V325h-4.047zm-25.516 0v-14.078h-2.53l1.562-4.625h10.89c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149V325h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V325h-4.047zm-9.656 0v-5.016h5.016V325h-5.016zm249.906 38v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-36.86 0l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.234.671c-.136.396-.203.74-.203 1.032 0 1.01.87 1.583 2.609 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.507 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.656 363h-14.5zm-20.656 0v-10.031h4.047V363H304.5zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-48.14 0l2.265-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L274.188 363h-4.235zm-7.376 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V363h-4.047zm-31.14 0v-14.078h-2.532l1.563-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V363h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.204V363h-4.047zm-6.344-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM198.204 363l1.546-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L214.405 363h-16.203zm-21.032 0v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625H187c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-46.97 0v-.516c0-2.677.282-4.971.845-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V363h-4.328zm-24.265 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 363h-8.313zm285.016 31.734h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM378.812 401l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.671 1.062 1.271 1.593 3.063 1.593 5.375 0 1.844-.49 4.032-1.468 6.563L391.5 401h-12.688zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-20.843.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-8.891-.39v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L322.188 401h-8.313zm-30.688 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.136-.281 1.932-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.719 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L275.75 401h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.197.792.296 2.177.296 4.156V401h-4.046v-9.063c0-2.416-.204-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L246.594 401h-8.313zm-31.234 0v-10.031h4.047V401h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047zm-39.72 0l1.548-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L189.234 401h-8.312zm-13.171.39l-1.594-3.859c2.334-1.135 4.724-2.724 7.172-4.765l-.39-2.485c-.23-1.427-.566-2.398-1.008-2.914-.443-.515-1.42-1.075-2.93-1.68l-.89-.359 1.75-3.406c2.562.844 4.317 1.687 5.265 2.531.625.563 1.055 1.177 1.29 1.844.233.667.517 2.094.85 4.281l.61 4c.448 2.938.89 5.078 1.328 6.422h-4.36c-.28-1.115-.577-2.594-.89-4.438-1.542 1.553-3.61 3.162-6.203 4.829zm-5.375-6.656h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM147.281 401v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 401h-8.313zm218.406 38v-10.031h4.047V439h-4.046zm13.594 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V439h-4.047zm-35.687 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L332.047 439h-16.203zm-18.61 0l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.454-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zm-11.125-1.64H285.5c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM263.375 439l-1.766-14.078h-2.515l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zm-11.125-1.64h-4.062c.729-2.667 1.093-4.574 1.093-5.72 0-.822-.359-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM224.781 439l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V439h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.094 439h-8.313zm-102.89 0v-23.125h4.812v9.39h9.563v-9.39h4.812V439h-4.812v-10.547h-9.563V439h-4.812zm30.703 0v-19.953h-8.281v-3.172h21.39v3.172h-8.281V439h-4.828zm16.344 0v-23.125h6.375l5.562 16.266 5.734-16.266h5.594V439h-4.437v-17.453l-5.641 15.906h-3.875l-5.531-16.14V439h-3.781zm29.046 0v-23.125h4.813v19.844h10.422V439h-15.235zm204.094 38v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V477h-4.047zm-25.625 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L380.688 477h-4.235zm-19.984 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V477h-4.328zm-20.328 0l1.39-4.234H348v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.781-.234.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.868v6.078L350.64 477h-14.5zm-20.11 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-9.969 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.046zm-33.375 0l1.547-4.625h4.032c-.823-2.115-1.235-4.708-1.235-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V477h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L281 477h-8.313zm-29.515 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L259.375 477h-16.203zm-18.156 0l1.53-4.625h8.735c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.958.312 1.77.87 2.437 1.671 1.062 1.271 1.594 3.063 1.594 5.375 0 1.844-.49 4.032-1.47 6.563L237.704 477h-12.687zm-7.547 4.625v-18.703h-10l1.547-4.625h12.5v23.328h-4.047zm-36.453-4.234l1.593-4.72c2.042-.228 3.688-.838 4.938-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.696-.484h-5.562v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-5.391-6.657h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM148.828 477l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-7.078 0v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V477h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.047zm207.406 31.734h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM297.656 515l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-10.047 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V515h-4.047zm-25.625 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L266.219 515h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 515v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V515h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 515l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L202.328 515h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.704c-2.01.958-4.849 1.437-8.515 1.437zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V515h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.454v7.187L149.86 515h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm212.14 38l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.531 6.687-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L338.75 553h-4.234zm-23.28 0l1.546-4.625h4.031c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.157V553h-4.046v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L319.547 553h-8.313zm-7.173 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.046zm-25.515 0v-14.078h-2.531l1.562-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V553h-4.047zm-20.485 0l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.672 1.062 1.27 1.593 3.062 1.593 5.375 0 1.843-.49 4.03-1.468 6.562L270.75 553h-12.688zm-21.156 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.53 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L241.14 553h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM204.641 553v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L220.453 553h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zm-20.953-1.64h-4.063c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM172.656 553v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L153.219 553h-8.313zm-7.094 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V553h-4.047zm-17.359 0v-5.016h5.016V553h-5.016zm322.688 38.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-22.594-.39l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.938 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.296-5.031-1.126 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L424.781 591h-4.234zm-21.625 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843L402.5 573.28l3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM380.078 591l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-23.562 0v-10.031h4.046V591h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591h-4.047zm-21.78 0h-1.923v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-31.204 0l1.531-4.625h8.735c.864-1.99 1.296-3.682 1.296-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.782 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L329.813 591h-12.688zm-3.953-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM298.078 591v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V591h-4.047zm-25.14 0v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L288.75 591h-15.813zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H277v9.453zM265.203 591h-1.922v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-20.484 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591h-4.047zm-17.547 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V591h-4.047zm-23.719 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L219.655 591h-16.203zm-30.594 0v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V591h-4.329zm-24.265 0l1.547-4.625h4.03c-.822-2.115-1.233-4.708-1.233-7.781v-1.672h-2.329l1.547-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591H164.5v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695H157v1.375c0 3.354.48 6.047 1.438 8.078L156.905 591h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.135 0 5.23.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591H141.5zm-14.047-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM406.016 629v-10.031h4.046V629h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-36.421 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V629h-4.329zm-4.938-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM356.078 629l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.704 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L365.656 629h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.848 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L313.188 629h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm-34.078 0v-10.031h4.047V629h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L231.86 629h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V629h-4.047zm-13.531 0v-14.078h-9.532l1.547-4.625h16.094l-1.547 4.625h-2.516V629h-4.046zm-16.282 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.516v-6.078c0-1.646-.355-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L175.312 629h-9.578zm-3.625-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.203 629v-14.078h-2.484l1.547-4.625h10.484c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L138.016 629h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.828-.687h-4.812v9.453zM401.844 667l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L406.078 667h-4.234zm-20.422 0l-2.203-17.5 3.922-1.578 1.171 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-18.969.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-20.687-.39l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L357.969 667h-16.203zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM309.437 667v-14.078h-2.484l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L325.25 667h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H313.5v9.453zM276.078 667l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-21.937 0l1.39-4.234H266v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.867v6.079L268.64 667h-14.5zm-20.657 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-38.234 0v-7.016c0-2.354.682-4.343 2.047-5.968L206 651.938l1.719-4.016 11.094 4.719-1.672 4.062-3.125-1.36c-.75.938-1.125 2.256-1.125 3.954V667h-4.047zm-9.735 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V667h-4.047zm-11.734 0v-5.016h5.016V667h-5.016zm-32.156 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-20.391-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM120.797 667v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V667h-4.328zm275.937 31.734h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM380.031 705v-14.078H370.5l1.547-4.625h16.094l-1.547 4.625h-2.516V705h-4.047zm-16.36 0v-9.594c0-1.927-.32-3.252-.96-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V705h-4.047zm-26.093 2.781l1.563-4.672c1.666 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.345v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zM318.188 705v-10.031h4.046V705h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm-46.234 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L301.75 705h-16.203zm-21.688 2.781l1.563-4.672c1.667 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.636-1.24 2.774-2.78 3.414-4.625.64-1.843.961-4.484.961-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zm-2.375-9.047h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.547-4.625h10.843c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.359 8.547-2.24 2.646-5.13 4.537-8.672 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.984-1.883-.657-.317-1.959-.476-3.907-.476h-10.39zM224.406 705v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V705h-4.047zm-38.734 0l1.531-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L198.359 705h-12.687zm-20.422 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.938 0l1.391-4.234h10.469v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.508 2.164.692.88 1.039 2.503 1.039 4.867v6.079L157.813 705h-14.5zm-20.656 0v-10.031h4.047V705h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.546-4.625h11.172c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm169.5 38v-10.031h4.047V743h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L285.984 743h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V743h-4.047zm-22.703 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-21.813 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L245.984 743h-16.203zm-21.687 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM186.39 743l1.547-4.625h4.03c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078l-1.53 4.625h-8.313zm-29.516 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L173.08 743h-16.204zm-9.125 0v-14.078h-9.531l1.547-4.625h16.093l-1.547 4.625h-2.515V743h-4.047zm-28.25 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.782-.234.672c-.136.396-.204.74-.204 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.867v6.079L134 743h-14.5z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M117 106h282v198H117z"/><g id="Rectangle-209-+-Rectangle-208" fill="#DBAF88" transform="translate(15 14)"><path id="Rectangle-209" d="M0 1h50v300H0z"/><path id="Rectangle-208" d="M165.5-129.5h50v311h-50z" transform="rotate(-90 190.5 26)"/></g><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M45-7h275v447H45z" transform="rotate(-180 182.5 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M319 8h162v399H319z"/><g id="Group-2" transform="translate(65 65)"><rect id="Rectangle-19" width="20" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-3)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="20" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(5.25 20)"><path id="Rectangle-22" d="M.5.5H10v1H.5z"/><path id="Rectangle-23" d="M.5 3.5H10v1H.5z"/><path id="Rectangle-24" d="M.5 6.5H10v1H.5z"/><path id="Rectangle-25" d="M.5 9.5H10v1H.5z"/></g></g></g><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M65.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H58.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M72 57.68l14 7-14 7-.001-6h-42.59l.001 6-14-7 14-7-.001 6h42.59l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="78.7" y="43" fill="#C06334">clientTop:</tspan> <tspan x="150.7" y="43" fill="#1C85B5">25px </tspan> <tspan x="186.7" y="43" fill="#C06334">= border</tspan></text><text id="clientLeft:41px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 40.32 125)"><tspan x="-13.68" y="129" fill="#C06334">clientLeft:</tspan> <tspan x="65.52" y="129" fill="#1C85B5">41px</tspan></text><g id="Rectangle-8-+-Rectangle-7" transform="translate(-42 -3)"><path id="Rectangle-8" fill="url(#linearGradient-1)" d="M86.5-85.5h275v447h-275z" transform="rotate(-90 224 138)"/><path id="Rectangle-7" fill="#FFF" d="M152 156h162v399H152z" transform="rotate(90 233 355.5)"/></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="359" height="316" viewBox="0 0 359 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top-rtl.svg"><path fill="#FFF" d="M0 0h359v316H0z"/><path id="היא-שפת-תסריט-מפורשת" fill="#643B0C" d="M336.484 135v-10.031h4.047V135h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047zm-20.39-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM302.063 135v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V135h-4.329zm-31.954 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.718 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L262.671 135h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.516 135h-8.313zm-33.547 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L199.969 135h-8.313zm-18.265 0l-1.766-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.876.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.343 5.156c-2.26 1.313-5.22 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.907-3.25 2.907-5.531 0-.938-.328-1.57-.985-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.157 9.453zM162.297 135v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.531-4.625h5.047c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.148V135h-4.047zm-14.047-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.375 135l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm222.344 38l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.757l.531 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L348.953 173h-4.234zm-19.344 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.874 173h-14.5zm-7.14 0v-9.594c0-1.927-.32-3.252-.962-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V173h-4.046zm-30.516 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zM250 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L258.313 173H250zm-31.422 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L222.813 173h-4.234zm-19.25 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L215.53 173h-16.203zm-7.516 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-23.015 0l-1.766-14.078h-2.515l1.547-4.625h11.078c2.093 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.105 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zM146.531 173l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.453-4.625c2.334-.177 4.089-.687 5.266-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zM119.531 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 173h-8.313zm288.766 38v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.665l-.703-.671c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V211h-4.328zm-8.438 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-23.718 0l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L392.344 211H376.14zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.594 0h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.546-4.625h10.844c1.99 0 3.39.088 4.203.266.813.177 1.495.541 2.047 1.093.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM312.594 211l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-6.516-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM277.766 211v-14.078h-2.485l1.547-4.625h10.485c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L293.578 211h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zM243.86 211v-10.031h4.047V211h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V211h-4.047zm-37.594 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L224.094 211h-4.235zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L192.969 211h-8.313zm-19.703 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.079-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.304-1.927.304-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.546 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V211h-4.329zm-22.14 0l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L147.047 211h-4.234zm-23.282 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 211h-8.313zm318.844 38.39l1.594-4.718c2.041-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.86-.39l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.28 2.312c2.72 0 4.891-.453 6.517-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-6.546-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zm-24.016 6.657l1.594-4.72c2.042-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-8.89-.391v-9.594c0-1.927-.32-3.252-.961-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V249h-4.046zm-23.72 0l1.548-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L369.547 249h-16.203zm-29.859 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.608-4.32-.407-.463-1.683-.695-3.829-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L339.688 249h-16.204zm-20.046 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-24.266 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.329l1.547-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V249h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L287.484 249h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-14.047-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 249v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 249l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.453v7.188L202.328 249h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L149.86 249h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm258.422 38v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.156V287h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V287h-4.047zm-29.25 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-20.594 0v-10.031h4.047V287h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V287h-4.047zm-48.61 2.781l1.563-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.265 14.563c1.636-1.24 2.774-2.78 3.415-4.625.64-1.843.96-4.484.96-7.921v-1.641h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zm-14.796-2.39l1.594-4.72c2.041-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.642-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-15.938-6.657h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM236.656 287v-14.078h-9.531l1.547-4.625h16.094l-1.547 4.625h-2.516V287h-4.047zm-12.86-6.266h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM196.547 287v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L182.796 287h-14.5zm-8.75 0v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L145.796 287h-14.5zm-7.155 4.625v-14.281c0-2.032-.36-3.344-1.079-3.938-1.041-.875-2.328-1.552-3.859-2.031l1.86-3.453c1.875.562 3.53 1.37 4.968 2.422.834.614 1.401 1.364 1.703 2.25.302.885.454 2.255.454 4.11v14.921h-4.047zM426.797 325v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.156V325h-4.046zm-36.422 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V325h-4.328zm-4.938-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM376.86 325l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.454v7.187L386.438 325h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm-9.937 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V325h-4.046zm-24.625 0l1.562-4.625h5.516v-6.078c0-1.646-.354-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.297 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L333.969 325h-9.578zm-17.907 0l-2.234-17.688 3.922-1.39 1.797 14.453c2.666 0 4.794-.615 6.383-1.844 1.588-1.229 2.382-2.87 2.382-4.922 0-2.24-1.208-3.359-3.625-3.359-.645 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.917 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437-1.395 2-3.291 3.568-5.687 4.704-2.01.958-4.849 1.437-8.516 1.437zm-32.359 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.197.797.296 2.18.296 4.149v6.562h2.5L290.33 325h-16.204zm-19.219 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.094-1.578a28.973 28.973 0 004.937 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-8.687 0v-14.078h-9.531l1.546-4.625h16.094l-1.547 4.625h-2.515V325h-4.047zm-40.422 0v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V325h-4.047zm-32.813.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.125 4.235v-14.61h4.047v14.61h-4.047zm-2.516-18.703l1.547-4.625h10.844c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM155.625 325v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V325h-4.047zm-25.516 0v-14.078h-2.53l1.562-4.625h10.89c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149V325h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V325h-4.047zm-9.656 0v-5.016h5.016V325h-5.016zm249.906 38v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-36.86 0l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.234.671c-.136.396-.203.74-.203 1.032 0 1.01.87 1.583 2.609 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.507 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.656 363h-14.5zm-20.656 0v-10.031h4.047V363H304.5zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-48.14 0l2.265-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L274.188 363h-4.235zm-7.376 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V363h-4.047zm-31.14 0v-14.078h-2.532l1.563-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V363h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.204V363h-4.047zm-6.344-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM198.204 363l1.546-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L214.405 363h-16.203zm-21.032 0v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625H187c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-46.97 0v-.516c0-2.677.282-4.971.845-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V363h-4.328zm-24.265 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 363h-8.313zm285.016 31.734h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM378.812 401l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.671 1.062 1.271 1.593 3.063 1.593 5.375 0 1.844-.49 4.032-1.468 6.563L391.5 401h-12.688zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-20.843.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-8.891-.39v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L322.188 401h-8.313zm-30.688 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.136-.281 1.932-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.719 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L275.75 401h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.197.792.296 2.177.296 4.156V401h-4.046v-9.063c0-2.416-.204-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L246.594 401h-8.313zm-31.234 0v-10.031h4.047V401h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047zm-39.72 0l1.548-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L189.234 401h-8.312zm-13.171.39l-1.594-3.859c2.334-1.135 4.724-2.724 7.172-4.765l-.39-2.485c-.23-1.427-.566-2.398-1.008-2.914-.443-.515-1.42-1.075-2.93-1.68l-.89-.359 1.75-3.406c2.562.844 4.317 1.687 5.265 2.531.625.563 1.055 1.177 1.29 1.844.233.667.517 2.094.85 4.281l.61 4c.448 2.938.89 5.078 1.328 6.422h-4.36c-.28-1.115-.577-2.594-.89-4.438-1.542 1.553-3.61 3.162-6.203 4.829zm-5.375-6.656h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM147.281 401v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 401h-8.313zm218.406 38v-10.031h4.047V439h-4.046zm13.594 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V439h-4.047zm-35.687 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L332.047 439h-16.203zm-18.61 0l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.454-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zm-11.125-1.64H285.5c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM263.375 439l-1.766-14.078h-2.515l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zm-11.125-1.64h-4.062c.729-2.667 1.093-4.574 1.093-5.72 0-.822-.359-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM224.781 439l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V439h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.094 439h-8.313zm-102.89 0v-23.125h4.812v9.39h9.563v-9.39h4.812V439h-4.812v-10.547h-9.563V439h-4.812zm30.703 0v-19.953h-8.281v-3.172h21.39v3.172h-8.281V439h-4.828zm16.344 0v-23.125h6.375l5.562 16.266 5.734-16.266h5.594V439h-4.437v-17.453l-5.641 15.906h-3.875l-5.531-16.14V439h-3.781zm29.046 0v-23.125h4.813v19.844h10.422V439h-15.235zm204.094 38v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V477h-4.047zm-25.625 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L380.688 477h-4.235zm-19.984 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V477h-4.328zm-20.328 0l1.39-4.234H348v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.781-.234.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.868v6.078L350.64 477h-14.5zm-20.11 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-9.969 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.046zm-33.375 0l1.547-4.625h4.032c-.823-2.115-1.235-4.708-1.235-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V477h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L281 477h-8.313zm-29.515 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L259.375 477h-16.203zm-18.156 0l1.53-4.625h8.735c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.958.312 1.77.87 2.437 1.671 1.062 1.271 1.594 3.063 1.594 5.375 0 1.844-.49 4.032-1.47 6.563L237.704 477h-12.687zm-7.547 4.625v-18.703h-10l1.547-4.625h12.5v23.328h-4.047zm-36.453-4.234l1.593-4.72c2.042-.228 3.688-.838 4.938-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.696-.484h-5.562v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-5.391-6.657h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM148.828 477l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-7.078 0v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V477h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.047zm207.406 31.734h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM297.656 515l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-10.047 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V515h-4.047zm-25.625 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L266.219 515h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 515v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V515h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 515l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L202.328 515h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.704c-2.01.958-4.849 1.437-8.515 1.437zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V515h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.454v7.187L149.86 515h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm212.14 38l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.531 6.687-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L338.75 553h-4.234zm-23.28 0l1.546-4.625h4.031c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.157V553h-4.046v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L319.547 553h-8.313zm-7.173 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.046zm-25.515 0v-14.078h-2.531l1.562-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V553h-4.047zm-20.485 0l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.672 1.062 1.27 1.593 3.062 1.593 5.375 0 1.843-.49 4.03-1.468 6.562L270.75 553h-12.688zm-21.156 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.53 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L241.14 553h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM204.641 553v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L220.453 553h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zm-20.953-1.64h-4.063c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM172.656 553v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L153.219 553h-8.313zm-7.094 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V553h-4.047zm-17.359 0v-5.016h5.016V553h-5.016zm322.688 38.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-22.594-.39l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.938 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.296-5.031-1.126 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L424.781 591h-4.234zm-21.625 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843L402.5 573.28l3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM380.078 591l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-23.562 0v-10.031h4.046V591h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591h-4.047zm-21.78 0h-1.923v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-31.204 0l1.531-4.625h8.735c.864-1.99 1.296-3.682 1.296-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.782 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L329.813 591h-12.688zm-3.953-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM298.078 591v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V591h-4.047zm-25.14 0v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L288.75 591h-15.813zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H277v9.453zM265.203 591h-1.922v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-20.484 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591h-4.047zm-17.547 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V591h-4.047zm-23.719 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L219.655 591h-16.203zm-30.594 0v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V591h-4.329zm-24.265 0l1.547-4.625h4.03c-.822-2.115-1.233-4.708-1.233-7.781v-1.672h-2.329l1.547-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591H164.5v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695H157v1.375c0 3.354.48 6.047 1.438 8.078L156.905 591h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.135 0 5.23.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591H141.5zm-14.047-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM406.016 629v-10.031h4.046V629h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-36.421 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V629h-4.329zm-4.938-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM356.078 629l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.704 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L365.656 629h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.848 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L313.188 629h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm-34.078 0v-10.031h4.047V629h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L231.86 629h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V629h-4.047zm-13.531 0v-14.078h-9.532l1.547-4.625h16.094l-1.547 4.625h-2.516V629h-4.046zm-16.282 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.516v-6.078c0-1.646-.355-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L175.312 629h-9.578zm-3.625-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.203 629v-14.078h-2.484l1.547-4.625h10.484c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L138.016 629h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.828-.687h-4.812v9.453zM401.844 667l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L406.078 667h-4.234zm-20.422 0l-2.203-17.5 3.922-1.578 1.171 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-18.969.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-20.687-.39l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L357.969 667h-16.203zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM309.437 667v-14.078h-2.484l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L325.25 667h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H313.5v9.453zM276.078 667l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-21.937 0l1.39-4.234H266v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.867v6.079L268.64 667h-14.5zm-20.657 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-38.234 0v-7.016c0-2.354.682-4.343 2.047-5.968L206 651.938l1.719-4.016 11.094 4.719-1.672 4.062-3.125-1.36c-.75.938-1.125 2.256-1.125 3.954V667h-4.047zm-9.735 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V667h-4.047zm-11.734 0v-5.016h5.016V667h-5.016zm-32.156 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-20.391-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM120.797 667v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V667h-4.328zm275.937 31.734h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM380.031 705v-14.078H370.5l1.547-4.625h16.094l-1.547 4.625h-2.516V705h-4.047zm-16.36 0v-9.594c0-1.927-.32-3.252-.96-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V705h-4.047zm-26.093 2.781l1.563-4.672c1.666 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.345v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zM318.188 705v-10.031h4.046V705h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm-46.234 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L301.75 705h-16.203zm-21.688 2.781l1.563-4.672c1.667 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.636-1.24 2.774-2.78 3.414-4.625.64-1.843.961-4.484.961-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zm-2.375-9.047h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.547-4.625h10.843c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.359 8.547-2.24 2.646-5.13 4.537-8.672 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.984-1.883-.657-.317-1.959-.476-3.907-.476h-10.39zM224.406 705v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V705h-4.047zm-38.734 0l1.531-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L198.359 705h-12.687zm-20.422 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.938 0l1.391-4.234h10.469v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.508 2.164.692.88 1.039 2.503 1.039 4.867v6.079L157.813 705h-14.5zm-20.656 0v-10.031h4.047V705h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.546-4.625h11.172c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm169.5 38v-10.031h4.047V743h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L285.984 743h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V743h-4.047zm-22.703 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-21.813 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L245.984 743h-16.203zm-21.687 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM186.39 743l1.547-4.625h4.03c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078l-1.53 4.625h-8.313zm-29.516 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L173.08 743h-16.204zm-9.125 0v-14.078h-9.531l1.547-4.625h16.093l-1.547 4.625h-2.515V743h-4.047zm-28.25 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.782-.234.672c-.136.396-.204.74-.204 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.867v6.079L134 743h-14.5z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M117 106h282v198H117z"/><g id="Rectangle-209-+-Rectangle-208" fill="#DBAF88" transform="translate(15 14)"><path id="Rectangle-209" d="M0 1h50v300H0z"/><path id="Rectangle-208" d="M165.5-129.5h50v311h-50z" transform="rotate(-90 190.5 26)"/></g><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M45-7h275v447H45z" transform="rotate(-180 182.5 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M319 8h162v399H319z"/><g id="Group-2" transform="translate(65 65)"><rect id="Rectangle-19" width="20" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-3)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="20" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(5.25 20)"><path id="Rectangle-22" d="M.5.5H10v1H.5z"/><path id="Rectangle-23" d="M.5 3.5H10v1H.5z"/><path id="Rectangle-24" d="M.5 6.5H10v1H.5z"/><path id="Rectangle-25" d="M.5 9.5H10v1H.5z"/></g></g></g><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M65.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H58.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M72 57.68l14 7-14 7-.001-6h-42.59l.001 6-14-7 14-7-.001 6h42.59l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="78.7" y="43" fill="#C06334">clientTop:</tspan> <tspan x="150.7" y="43" fill="#1C85B5">25px </tspan> <tspan x="186.7" y="43" fill="#C06334">= border</tspan></text><text id="clientLeft:41px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 40.32 125)"><tspan x="-13.68" y="129" fill="#C06334">clientLeft:</tspan> <tspan x="65.52" y="129" fill="#1C85B5">41px</tspan></text><g id="Rectangle-8-+-Rectangle-7" transform="translate(-42 -3)"><path id="Rectangle-8" fill="url(#linearGradient-1)" d="M86.5-85.5h275v447h-275z" transform="rotate(-90 224 138)"/><path id="Rectangle-7" fill="#FFF" d="M152 156h162v399H152z" transform="rotate(90 233 355.5)"/></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="359" height="316" viewBox="0 0 359 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF" stop-opacity="0"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-4" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF" stop-opacity="0"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top-rtl.svg"><path fill="#FFF" d="M0 0h359v316H0z"/><path id="היא-שפת-תסריט-מפורשת" fill="#643B0C" d="M336.484 135v-10.031h4.047V135h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047zm-20.39-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM302.063 135v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V135h-4.329zm-31.954 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.718 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L262.671 135h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.516 135h-8.313zm-33.547 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V135h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L199.969 135h-8.313zm-18.265 0l-1.766-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.876.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.343 5.156c-2.26 1.313-5.22 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.907-3.25 2.907-5.531 0-.938-.328-1.57-.985-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.157 9.453zM162.297 135v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.531-4.625h5.047c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.148V135h-4.047zm-14.047-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.375 135l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm222.344 38l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.757l.531 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L348.953 173h-4.234zm-19.344 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.874 173h-14.5zm-7.14 0v-9.594c0-1.927-.32-3.252-.962-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V173h-4.046zm-30.516 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zM250 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L258.313 173H250zm-31.422 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L222.813 173h-4.234zm-19.25 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L215.53 173h-16.203zm-7.516 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V173h-4.047zm-23.015 0l-1.766-14.078h-2.515l1.547-4.625h11.078c2.093 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.105 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zM146.531 173l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.453-4.625c2.334-.177 4.089-.687 5.266-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zM119.531 173l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V173h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 173h-8.313zm288.766 38v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.665l-.703-.671c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V211h-4.328zm-8.438 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-23.718 0l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L392.344 211H376.14zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.594 0h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.546-4.625h10.844c1.99 0 3.39.088 4.203.266.813.177 1.495.541 2.047 1.093.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM312.594 211l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-6.516-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM277.766 211v-14.078h-2.485l1.547-4.625h10.485c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L293.578 211h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zM243.86 211v-10.031h4.047V211h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V211h-4.047zm-37.594 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L224.094 211h-4.235zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V211h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L192.969 211h-8.313zm-19.703 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.079-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.304-1.927.304-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.546 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V211h-4.329zm-22.14 0l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L147.047 211h-4.234zm-23.282 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V211h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 211h-8.313zm318.844 38.39l1.594-4.718c2.041-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.86-.39l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.28 2.312c2.72 0 4.891-.453 6.517-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-6.546-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zm-24.016 6.657l1.594-4.72c2.042-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-8.89-.391v-9.594c0-1.927-.32-3.252-.961-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V249h-4.046zm-23.72 0l1.548-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L369.547 249h-16.203zm-29.859 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.608-4.32-.407-.463-1.683-.695-3.829-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L339.688 249h-16.204zm-20.046 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-24.266 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.329l1.547-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V249h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L287.484 249h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-14.047-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 249v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.665l-.703-.671c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V249h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 249l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.453v7.188L202.328 249h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V249h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.234-.948-.678-2.224-1.245-3.828-1.704l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L149.86 249h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm258.422 38v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.156V287h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V287h-4.047zm-29.25 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-20.594 0v-10.031h4.047V287h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V287h-4.047zm-48.61 2.781l1.563-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.265 14.563c1.636-1.24 2.774-2.78 3.415-4.625.64-1.843.96-4.484.96-7.921v-1.641h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zm-14.796-2.39l1.594-4.72c2.041-.228 3.687-.838 4.937-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.642-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-15.938-6.657h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM236.656 287v-14.078h-9.531l1.547-4.625h16.094l-1.547 4.625h-2.516V287h-4.047zm-12.86-6.266h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM196.547 287v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L182.796 287h-14.5zm-8.75 0v-14.078h-9.53l1.547-4.625h16.093l-1.547 4.625h-2.515V287h-4.047zm-28.25 0l1.392-4.234h10.468v-4.157c0-1.948-.192-3.205-.578-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.374-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L145.796 287h-14.5zm-7.155 4.625v-14.281c0-2.032-.36-3.344-1.079-3.938-1.041-.875-2.328-1.552-3.859-2.031l1.86-3.453c1.875.562 3.53 1.37 4.968 2.422.834.614 1.401 1.364 1.703 2.25.302.885.454 2.255.454 4.11v14.921h-4.047zM426.797 325v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.156V325h-4.046zm-36.422 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V325h-4.328zm-4.938-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM376.86 325l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.454v7.187L386.438 325h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm-9.937 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.198.797.296 2.18.296 4.149V325h-4.046zm-24.625 0l1.562-4.625h5.516v-6.078c0-1.646-.354-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.297 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L333.969 325h-9.578zm-17.907 0l-2.234-17.688 3.922-1.39 1.797 14.453c2.666 0 4.794-.615 6.383-1.844 1.588-1.229 2.382-2.87 2.382-4.922 0-2.24-1.208-3.359-3.625-3.359-.645 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.917 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437-1.395 2-3.291 3.568-5.687 4.704-2.01.958-4.849 1.437-8.516 1.437zm-32.359 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.197.797.296 2.18.296 4.149v6.562h2.5L290.33 325h-16.204zm-19.219 0l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.094-1.578a28.973 28.973 0 004.937 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-8.687 0v-14.078h-9.531l1.546-4.625h16.094l-1.547 4.625h-2.515V325h-4.047zm-40.422 0v-10.031h4.047V325h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V325h-4.047zm-32.813.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-21.125 4.235v-14.61h4.047v14.61h-4.047zm-2.516-18.703l1.547-4.625h10.844c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.36 8.547-2.239 2.646-5.13 4.537-8.671 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.985-1.883-.656-.317-1.958-.476-3.906-.476h-10.39zM155.625 325v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V325h-4.047zm-25.516 0v-14.078h-2.53l1.562-4.625h10.89c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149V325h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V325h-4.047zm-9.656 0v-5.016h5.016V325h-5.016zm249.906 38v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.171c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-36.86 0l-2.202-17.5 3.921-1.578 1.172 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.993l.281 2.312c2.719 0 4.89-.453 6.516-1.36 2.843-1.583 4.265-5.192 4.265-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.234.671c-.136.396-.203.74-.203 1.032 0 1.01.87 1.583 2.609 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.507 2.164.693.88 1.04 2.503 1.04 4.868v6.078L339.656 363h-14.5zm-20.656 0v-10.031h4.047V363H304.5zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-48.14 0l2.265-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L274.188 363h-4.235zm-7.376 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V363h-4.047zm-31.14 0v-14.078h-2.532l1.563-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V363h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.204V363h-4.047zm-6.344-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM198.204 363l1.546-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L214.405 363h-16.203zm-21.032 0v-10.031h4.047V363h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625H187c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V363h-4.047zm-46.97 0v-.516c0-2.677.282-4.971.845-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V363h-4.328zm-24.265 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V363h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 363h-8.313zm285.016 31.734h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM378.812 401l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.671 1.062 1.271 1.593 3.063 1.593 5.375 0 1.844-.49 4.032-1.468 6.563L391.5 401h-12.688zm-7.453 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-20.843.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-8.891-.39v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.579c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L322.188 401h-8.313zm-30.688 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.136-.281 1.932-.945 2.39-1.992.46-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.368 2.993l.282 2.312c2.719 0 4.89-.453 6.515-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.794 4.359-13.899 4.359zm-21.937 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.781-.235.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.868v6.078L275.75 401h-14.5zm-22.969 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.197.792.296 2.177.296 4.156V401h-4.046v-9.063c0-2.416-.204-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L246.594 401h-8.313zm-31.234 0v-10.031h4.047V401h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.684-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V401h-4.047zm-39.72 0l1.548-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.546-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L189.234 401h-8.312zm-13.171.39l-1.594-3.859c2.334-1.135 4.724-2.724 7.172-4.765l-.39-2.485c-.23-1.427-.566-2.398-1.008-2.914-.443-.515-1.42-1.075-2.93-1.68l-.89-.359 1.75-3.406c2.562.844 4.317 1.687 5.265 2.531.625.563 1.055 1.177 1.29 1.844.233.667.517 2.094.85 4.281l.61 4c.448 2.938.89 5.078 1.328 6.422h-4.36c-.28-1.115-.577-2.594-.89-4.438-1.542 1.553-3.61 3.162-6.203 4.829zm-5.375-6.656h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM147.281 401v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V401h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V401h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L127.844 401h-8.313zm218.406 38v-10.031h4.047V439h-4.046zm13.594 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V439h-4.047zm-35.687 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.828l1.546-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L332.047 439h-16.203zm-18.61 0l-1.765-14.078h-2.516l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.313 2.047 1.313 3.765 0 2.292-.662 4.506-1.985 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.218 1.969-8.875 1.969zm3.454-4.625c2.333-.177 4.088-.687 5.265-1.531 1.938-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.656-.328-1.948-.492-3.875-.492h-4.469l1.156 9.453zm-11.125-1.64H285.5c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM263.375 439l-1.766-14.078h-2.515l1.547-4.625h11.078c2.094 0 3.578.078 4.453.234.875.157 1.573.469 2.094.938.875.791 1.312 2.047 1.312 3.765 0 2.292-.661 4.506-1.984 6.641s-3.104 3.854-5.344 5.156c-2.26 1.313-5.219 1.969-8.875 1.969zm3.453-4.625c2.333-.177 4.089-.687 5.266-1.531 1.937-1.407 2.906-3.25 2.906-5.531 0-.938-.328-1.57-.984-1.899-.657-.328-1.948-.492-3.875-.492h-4.47l1.157 9.453zm-11.125-1.64h-4.062c.729-2.667 1.093-4.574 1.093-5.72 0-.822-.359-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM224.781 439l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.156V439h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L233.094 439h-8.313zm-102.89 0v-23.125h4.812v9.39h9.563v-9.39h4.812V439h-4.812v-10.547h-9.563V439h-4.812zm30.703 0v-19.953h-8.281v-3.172h21.39v3.172h-8.281V439h-4.828zm16.344 0v-23.125h6.375l5.562 16.266 5.734-16.266h5.594V439h-4.437v-17.453l-5.641 15.906h-3.875l-5.531-16.14V439h-3.781zm29.046 0v-23.125h4.813v19.844h10.422V439h-15.235zm204.094 38v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V477h-4.047zm-25.625 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.53 6.688-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.195 5.664L380.688 477h-4.235zm-19.984 0v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.447 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V477h-4.328zm-20.328 0l1.39-4.234H348v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.781-.234.671c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.046 0-.917.375-2.438 1.125-4.563l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.868v6.078L350.64 477h-14.5zm-20.11 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-9.969 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.046zm-33.375 0l1.547-4.625h4.032c-.823-2.115-1.235-4.708-1.235-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.156V477h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L281 477h-8.313zm-29.515 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L259.375 477h-16.203zm-18.156 0l1.53-4.625h8.735c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.958.312 1.77.87 2.437 1.671 1.062 1.271 1.594 3.063 1.594 5.375 0 1.844-.49 4.032-1.47 6.563L237.704 477h-12.687zm-7.547 4.625v-18.703h-10l1.547-4.625h12.5v23.328h-4.047zm-36.453-4.234l1.593-4.72c2.042-.228 3.688-.838 4.938-1.827 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.696-.484h-5.562v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-5.391-6.657h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM148.828 477l1.36-4.14h8.468c-3.833-3.709-6.713-8.162-8.64-13.36l4.093-1.578a28.973 28.973 0 004.938 8.875c1.469-1.781 2.203-3.943 2.203-6.485v-2.015h4.344v1.312c0 3.97-1.36 7.214-4.078 9.735 1.573 1.614 2.974 2.791 4.203 3.531l-1.39 4.125h-15.5zm-7.078 0v-9.594c0-1.927-.32-3.252-.96-3.976-.642-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.244.864.367 2.25.367 4.156V477h-4.047zm-11.844 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V477h-4.047zm207.406 31.734h-4.062c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.72-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.735-1.25l2.094-3.89c2.552.51 4.331 1.21 5.336 2.101s1.508 2.206 1.508 3.946c0 1.094-.25 2.76-.75 5l-.406 1.765zM297.656 515l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.993l.281 2.312c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.58 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-10.047 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.156V515h-4.047zm-25.625 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.757l.531 6.688-1.531 4.625h-9.031l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L266.219 515h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.954-.718-.479-1.963-.895-3.734-1.25l2.094-3.89c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM219.86 515v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V515h-4.329zm-4.937-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.954-.719-.479-1.964-.895-3.734-1.25l2.093-3.89c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM192.75 515l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.703 1.86.907.53 1.532 1.182 1.875 1.952.344.771.516 1.922.516 3.454v7.187L202.328 515h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.704c-2.01.958-4.849 1.437-8.515 1.437zm-9.938 0v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V515h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.454v7.187L149.86 515h-9.578zm-17.906 0l-2.234-17.688 3.921-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.415.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.687 4.704c-2.01.958-4.85 1.437-8.516 1.437zm212.14 38l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.531 6.687-1.531 4.625h-9.031l1.547-4.625h4.921l-.359-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L338.75 553h-4.234zm-23.28 0l1.546-4.625h4.031c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.296 2.177.296 4.157V553h-4.046v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078L319.547 553h-8.313zm-7.173 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.208 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.046zm-25.515 0v-14.078h-2.531l1.562-4.625h10.89c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.203V553h-4.047zm-20.485 0l1.532-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.761-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.15.156 4.11.469.957.312 1.77.87 2.437 1.672 1.062 1.27 1.593 3.062 1.593 5.375 0 1.843-.49 4.03-1.468 6.562L270.75 553h-12.688zm-21.156 0l2.266-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.219 2.008 1.052 1.338 1.677 3.257 1.875 5.758l.53 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.497 3.007-2.195 5.664L241.14 553h-4.235zm-3.953-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM204.641 553v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L220.453 553h-15.812zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687h-4.813v9.453zm-20.953-1.64h-4.063c.73-2.667 1.094-4.574 1.094-5.72 0-.822-.36-1.473-1.078-1.952-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM172.656 553v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V553h-4.047zm-27.75 0l1.547-4.625h4.031c-.823-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V553h-4.047v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-3.063v1.375c0 3.354.48 6.047 1.438 8.078L153.219 553h-8.313zm-7.094 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.28 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V553h-4.047zm-17.359 0v-5.016h5.016V553h-5.016zm322.688 38.39l1.593-4.718c2.042-.23 3.688-.839 4.938-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.961-1.844-.64-.323-1.873-.484-3.695-.484h-5.563v-9.625l4.047-1.36v6.36h4.062c2.5 0 4.243.36 5.227 1.078.984.719 1.476 1.984 1.476 3.797 0 3.437-1.317 6.49-3.953 9.156-2.635 2.667-6 4.354-10.093 5.063zm-22.594-.39l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.938 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.296-5.031-1.126 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L424.781 591h-4.234zm-21.625 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843L402.5 573.28l3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM380.078 591l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-23.562 0v-10.031h4.046V591h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591h-4.047zm-21.78 0h-1.923v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-31.204 0l1.531-4.625h8.735c.864-1.99 1.296-3.682 1.296-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.782 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L329.813 591h-12.688zm-3.953-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM298.078 591v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.071-1.711l2-3.797c2.448.448 4.208 1.15 5.281 2.11.708.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V591h-4.047zm-25.14 0v-14.078h-2.485l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L288.75 591h-15.813zm4.062-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H277v9.453zM265.203 591h-1.922v-5.016h5.016v3.891c0 4.094-1.672 6.14-5.016 6.14v-1.734c1.282 0 1.922-.927 1.922-2.781v-.5zm-20.484 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.048c3.135 0 5.229.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591h-4.047zm-17.547 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V591h-4.047zm-23.719 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L219.655 591h-16.203zm-30.594 0v-.516c0-2.677.282-4.971.844-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.183 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V591h-4.329zm-24.265 0l1.547-4.625h4.03c-.822-2.115-1.233-4.708-1.233-7.781v-1.672h-2.329l1.547-4.625h8.578c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V591H164.5v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695H157v1.375c0 3.354.48 6.047 1.438 8.078L156.905 591h-8.312zm-7.094 0v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.891l1.531-4.625h5.047c3.135 0 5.23.458 6.281 1.375a3.773 3.773 0 011.219 1.992c.198.797.297 2.18.297 4.149V591H141.5zm-14.047-6.266h-4.062c.729-2.666 1.093-4.573 1.093-5.718 0-.823-.359-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.507 2.206 1.507 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM406.016 629v-10.031h4.046V629h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-36.421 0v-.516c0-2.677.28-4.971.843-6.882.563-1.912 1.51-3.8 2.844-5.664l-.703-.672c-1.136-1.063-2.162-1.948-3.078-2.657l3.687-2.687c2.448 1.937 5.037 4.51 7.766 7.719.625-1.084 1.039-2.032 1.242-2.844.203-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.485 2.578c-2.062-3.709-4.864-7.355-8.406-10.938-1.26 1.958-1.89 4.948-1.89 8.969V629h-4.329zm-4.938-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM356.078 629l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.843-3.437c1.73.427 3.297 1.047 4.704 1.86.906.53 1.53 1.182 1.875 1.952.343.771.515 1.922.515 3.453v7.188L365.656 629h-9.578zm-17.906 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.493 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.848 1.438-8.515 1.438zm-9.938 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.515v-6.078c0-1.646-.354-2.724-1.062-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L313.188 629h-9.579zm-17.906 0l-2.234-17.688 3.922-1.39 1.796 14.453c2.667 0 4.795-.615 6.383-1.844 1.589-1.229 2.383-2.87 2.383-4.922 0-2.24-1.208-3.359-3.625-3.359-.646 0-1.338.104-2.078.313l1.406-4.36a9.102 9.102 0 012.235-.297c1.916 0 3.414.576 4.492 1.727 1.078 1.15 1.617 2.747 1.617 4.789 0 2.292-.698 4.437-2.094 6.437s-3.291 3.568-5.687 4.703c-2.01.959-4.85 1.438-8.516 1.438zm-34.078 0v-10.031h4.047V629h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V629h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L231.86 629h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V629h-4.047zm-13.531 0v-14.078h-9.532l1.547-4.625h16.094l-1.547 4.625h-2.516V629h-4.046zm-16.282 0v-9.063c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.282 1.375a3.773 3.773 0 011.218 1.992c.198.797.297 2.18.297 4.149V629h-4.047zm-24.625 0l1.563-4.625h5.516v-6.078c0-1.646-.355-2.724-1.063-3.235-.948-.677-2.224-1.244-3.828-1.703l1.844-3.437c1.729.427 3.296 1.047 4.703 1.86.906.53 1.531 1.182 1.875 1.952.344.771.515 1.922.515 3.453v7.188L175.312 629h-9.578zm-3.625-6.266h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.72-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-11.593 0h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.735-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM122.203 629v-14.078h-2.484l1.547-4.625h10.484c3.135 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562L138.016 629h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.828-.687h-4.812v9.453zM401.844 667l2.265-8.688c.417-1.593.625-2.692.625-3.296 0-1.24-1.057-2.907-3.171-5l3.609-2.094c1.146.958 2.125 2.265 2.937 3.922 1.667-2.615 3.667-3.922 6-3.922 1.76 0 3.167.67 4.22 2.008 1.051 1.338 1.676 3.257 1.874 5.758l.531 6.687-1.53 4.625h-9.032l1.547-4.625h4.922l-.36-4.797c-.25-3.354-1.349-5.031-3.297-5.031-1.125 0-2.07.56-2.836 1.68-.765 1.12-1.497 3.007-2.195 5.664L406.078 667h-4.234zm-20.422 0l-2.203-17.5 3.922-1.578 1.171 9.25c1.136-.281 1.933-.945 2.391-1.992.458-1.047.688-2.716.688-5.008v-1.875h4.343v1.719c0 3.093-.565 5.445-1.695 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.265-5.192 4.265-10.827v-1.891h4.329v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-18.969.39l1.594-4.718c2.042-.23 3.687-.839 4.937-1.828 1.948-1.542 2.922-3.406 2.922-5.594 0-.906-.32-1.52-.96-1.844-.641-.323-1.873-.484-3.696-.484h-5.563v-9.625l4.047-1.36v6.36h4.063c2.5 0 4.242.36 5.226 1.078.985.719 1.477 1.984 1.477 3.797 0 3.437-1.318 6.49-3.953 9.156-2.636 2.667-6 4.354-10.094 5.063zm-20.687-.39l1.546-4.625h9.641v-4.438c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.827-.695h-5.829l1.547-4.625h4.954c3.135 0 5.229.458 6.28 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L357.969 667h-16.203zm-4.016-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.335 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM309.437 667v-14.078h-2.484l1.547-4.625h10.484c3.136 0 5.23.458 6.282 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562L325.25 667h-15.813zm4.063-4.625h9.25v-4.438c0-2.427-.203-3.87-.61-4.328-.406-.458-1.682-.687-3.827-.687H313.5v9.453zM276.078 667l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.565 5.445-1.696 7.054-1.13 1.61-2.919 2.607-5.367 2.992l.281 2.313c2.72 0 4.891-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.742 12.782-3.162 2.906-7.795 4.359-13.899 4.359zm-21.937 0l1.39-4.234H266v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.039 2.503 1.039 4.867v6.079L268.64 667h-14.5zm-20.657 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-38.234 0v-7.016c0-2.354.682-4.343 2.047-5.968L206 651.938l1.719-4.016 11.094 4.719-1.672 4.062-3.125-1.36c-.75.938-1.125 2.256-1.125 3.954V667h-4.047zm-9.735 0v-9.594c0-1.927-.32-3.252-.96-3.976-.641-.724-1.998-1.295-4.07-1.711l2-3.797c2.447.448 4.207 1.15 5.28 2.11.709.635 1.185 1.385 1.43 2.25.245.864.367 2.25.367 4.155V667h-4.047zm-11.734 0v-5.016h5.016V667h-5.016zm-32.156 0v-10.031h4.047V667h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.618-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.281 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V667h-4.047zm-20.391-6.266h-4.063c.73-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.078-1.953-.719-.48-1.964-.896-3.734-1.25l2.093-3.891c2.552.51 4.331 1.21 5.336 2.101 1.006.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.406 1.765zM120.797 667v-.516c0-2.677.281-4.971.844-6.882.562-1.912 1.51-3.8 2.843-5.664l-.703-.672c-1.135-1.063-2.161-1.948-3.078-2.657l3.688-2.687c2.448 1.937 5.036 4.51 7.765 7.719.625-1.084 1.04-2.032 1.242-2.844.204-.813.305-1.927.305-3.344v-1.156h4.328v.906c0 3.521-1.182 6.568-3.547 9.14a41.618 41.618 0 014.422 6.47l-3.484 2.578c-2.063-3.709-4.865-7.355-8.406-10.938-1.26 1.958-1.891 4.948-1.891 8.969V667h-4.328zm275.937 31.734h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zM380.031 705v-14.078H370.5l1.547-4.625h16.094l-1.547 4.625h-2.516V705h-4.047zm-16.36 0v-9.594c0-1.927-.32-3.252-.96-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V705h-4.047zm-26.093 2.781l1.563-4.672c1.666 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.345v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zM318.188 705v-10.031h4.046V705h-4.046zm13.593 0v-9.063c0-2.416-.205-3.856-.617-4.32-.411-.463-1.685-.695-3.82-.695h-12.047l1.547-4.625h11.172c3.135 0 5.229.458 6.28 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm-46.234 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L301.75 705h-16.203zm-21.688 2.781l1.563-4.672c1.667 0 3.458-.28 5.375-.843l-3.36-14.985 3.875-1.36 3.266 14.563c1.636-1.24 2.774-2.78 3.414-4.625.64-1.843.961-4.484.961-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.154 4.88-5.79 6.64-3.697 2.469-7.635 3.703-11.812 3.703zm-2.375-9.047h-4.062c.729-2.666 1.094-4.573 1.094-5.718 0-.823-.36-1.474-1.079-1.953-.718-.48-1.963-.896-3.734-1.25l2.094-3.891c2.552.51 4.33 1.21 5.336 2.101 1.005.891 1.508 2.206 1.508 3.946 0 1.094-.25 2.76-.75 5l-.407 1.765zm-27.328 10.891v-14.61h4.047v14.61h-4.047zm-2.515-18.703l1.547-4.625h10.843c1.99 0 3.39.088 4.203.265.813.178 1.495.542 2.047 1.094.802.813 1.203 1.98 1.203 3.5 0 3.052-1.12 5.901-3.359 8.547-2.24 2.646-5.13 4.537-8.672 5.672l1.672-4.969c1.656-.469 3.036-1.383 4.14-2.742 1.105-1.36 1.657-2.82 1.657-4.383 0-.937-.328-1.565-.984-1.883-.657-.317-1.959-.476-3.907-.476h-10.39zM224.406 705v-9.063c0-2.416-.203-3.856-.61-4.32-.405-.463-1.681-.695-3.827-.695h-5.89l1.53-4.625h5.047c3.136 0 5.23.458 6.281 1.375a3.773 3.773 0 011.22 1.992c.197.797.296 2.18.296 4.149V705h-4.047zm-38.734 0l1.531-4.625h8.734c.865-1.99 1.297-3.682 1.297-5.078 0-1.646-.38-2.787-1.14-3.422-.76-.635-2.13-.953-4.11-.953h-5.406l1.531-4.625h5.094c1.781 0 3.151.156 4.11.469.958.312 1.77.87 2.437 1.672 1.063 1.27 1.594 3.062 1.594 5.375 0 1.843-.49 4.03-1.469 6.562L198.359 705h-12.687zm-20.422 0l-2.203-17.5 3.922-1.578 1.172 9.25c1.135-.281 1.932-.945 2.39-1.992.459-1.047.688-2.716.688-5.008v-1.875h4.344v1.719c0 3.093-.566 5.445-1.696 7.054-1.13 1.61-2.92 2.607-5.367 2.992l.281 2.313c2.719 0 4.89-.453 6.516-1.36 2.844-1.583 4.266-5.192 4.266-10.827v-1.891h4.328v1.562c0 5.615-1.581 9.875-4.743 12.782-3.161 2.906-7.794 4.359-13.898 4.359zm-21.938 0l1.391-4.234h10.469v-4.157c0-1.948-.193-3.205-.578-3.773-.386-.568-1.386-1.065-3-1.492l-2.985-.782-.234.672c-.135.396-.203.74-.203 1.032 0 1.01.87 1.583 2.61 1.718l-1.079 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.969-2.703 7.125 1.734c2.312.563 3.815 1.284 4.508 2.164.692.88 1.039 2.503 1.039 4.867v6.079L157.813 705h-14.5zm-20.656 0v-10.031h4.047V705h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.047l1.546-4.625h11.172c3.136 0 5.23.458 6.282 1.375a3.733 3.733 0 011.234 1.984c.198.792.297 2.177.297 4.157V705h-4.047zm169.5 38v-10.031h4.047V743h-4.047zm13.594 0v-9.063c0-2.416-.206-3.856-.617-4.32-.412-.463-1.685-.695-3.82-.695h-12.048l1.547-4.625h11.172c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047zm-37.594 0l2.266-8.688c.416-1.593.625-2.692.625-3.296 0-1.24-1.058-2.907-3.172-5l3.61-2.094c1.145.958 2.124 2.265 2.937 3.922 1.666-2.615 3.666-3.922 6-3.922 1.76 0 3.166.67 4.218 2.008s1.677 3.257 1.875 5.758l.532 6.687-1.532 4.625h-9.03l1.546-4.625h4.922l-.36-4.797c-.25-3.354-1.348-5.031-3.296-5.031-1.125 0-2.07.56-2.836 1.68-.766 1.12-1.498 3.007-2.196 5.664L285.984 743h-4.234zm-7.453 0v-9.594c0-1.927-.32-3.252-.961-3.976-.64-.724-1.997-1.295-4.07-1.711l2-3.797c2.448.448 4.208 1.15 5.28 2.11.71.635 1.186 1.385 1.43 2.25.245.864.368 2.25.368 4.155V743h-4.047zm-22.703 0l-2.235-17.688 3.922-1.39 1.797 14.453c2.667 0 4.794-.615 6.383-1.844 1.588-1.229 2.383-2.87 2.383-4.922 0-2.24-1.209-3.359-3.625-3.359-.646 0-1.339.104-2.078.313l1.406-4.36a9.102 9.102 0 012.234-.297c1.917 0 3.414.576 4.492 1.727 1.079 1.15 1.618 2.747 1.618 4.789 0 2.292-.698 4.437-2.094 6.437s-3.292 3.568-5.688 4.703c-2.01.959-4.849 1.438-8.515 1.438zm-21.813 0l1.547-4.625h9.64v-4.438c0-2.416-.202-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.136 0 5.23.458 6.281 1.375a3.734 3.734 0 011.235 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L245.984 743h-16.203zm-21.687 2.781l1.562-4.672c1.667 0 3.459-.28 5.375-.843l-3.36-14.985 3.876-1.36 3.266 14.563c1.635-1.24 2.773-2.78 3.414-4.625.64-1.843.96-4.484.96-7.922v-1.64h4.344v1.156c0 3.99-.612 7.318-1.836 9.985-1.224 2.666-3.153 4.88-5.789 6.64-3.698 2.469-7.635 3.703-11.812 3.703zM186.39 743l1.547-4.625h4.03c-.822-2.115-1.234-4.708-1.234-7.781v-1.672h-2.328l1.547-4.625h8.578c3.136 0 5.23.458 6.281 1.375a3.733 3.733 0 011.235 1.984c.198.792.297 2.177.297 4.157V743h-4.047v-9.063c0-2.416-.203-3.856-.61-4.32-.406-.463-1.682-.695-3.828-.695h-3.062v1.375c0 3.354.479 6.047 1.437 8.078l-1.53 4.625h-8.313zm-29.516 0l1.547-4.625h9.64v-4.438c0-2.416-.203-3.856-.609-4.32-.406-.463-1.682-.695-3.828-.695h-5.828l1.547-4.625h4.953c3.135 0 5.229.458 6.281 1.375a3.734 3.734 0 011.234 1.992c.198.797.297 2.18.297 4.149v6.562h2.5L173.08 743h-16.204zm-9.125 0v-14.078h-9.531l1.547-4.625h16.093l-1.547 4.625h-2.515V743h-4.047zm-28.25 0l1.39-4.234h10.47v-4.157c0-1.948-.193-3.205-.579-3.773-.385-.568-1.385-1.065-3-1.492l-2.984-.782-.234.672c-.136.396-.204.74-.204 1.032 0 1.01.87 1.583 2.61 1.718l-1.078 3.25c-3.5-.323-5.25-1.671-5.25-4.047 0-.916.375-2.437 1.125-4.562l.968-2.703 7.125 1.734c2.313.563 3.815 1.284 4.508 2.164.693.88 1.04 2.503 1.04 4.867v6.079L134 743h-14.5z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M117 106h282v198H117z"/><g id="Rectangle-209-+-Rectangle-208" fill="#DBAF88" transform="translate(15 15)"><path id="Rectangle-209" d="M0 0h50v300H0z"/><path id="Rectangle-208" d="M165.5-130.5h50v311h-50z" transform="rotate(-90 190.5 25)"/></g><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M158-7h162v447H158z" transform="rotate(-180 239 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M319 8h162v399H319z"/><g id="Group-2" transform="translate(65 65)"><rect id="Rectangle-19" width="20" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="20" height="19" x=".5" y=".5" fill="url(#linearGradient-2)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M10.5 7l4.2 6H6.3z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-3)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="20" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(5.25 20)"><path id="Rectangle-22" d="M.5.5H10v1H.5z"/><path id="Rectangle-23" d="M.5 3.5H10v1H.5z"/><path id="Rectangle-24" d="M.5 6.5H10v1H.5z"/><path id="Rectangle-25" d="M.5 9.5H10v1H.5z"/></g></g></g><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M65.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H58.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M72 57.68l14 7-14 7-.001-6h-42.59l.001 6-14-7 14-7-.001 6h42.59l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="78.7" y="43" fill="#643B0C">clientTop:</tspan> <tspan x="150.7" y="43" fill="#166388">25px</tspan> <tspan x="179.5" y="43" fill="#1C85B5"> </tspan> <tspan x="186.7" y="43" fill="#643B0C">= border</tspan></text><text id="clientLeft:41px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 40.32 125)"><tspan x="-13.68" y="129" fill="#643B0C">clientLeft:</tspan> <tspan x="65.52" y="129" fill="#166388">41px</tspan></text><g id="Rectangle-8-+-Rectangle-7" transform="translate(-41.5 128.5)"><path id="Rectangle-8" fill="url(#linearGradient-4)" d="M151.5-151.5h144v447h-144z" transform="rotate(-90 223.5 72)"/><path id="Rectangle-7" fill="#FFF" d="M151.5 24.5h162v399h-162z" transform="rotate(90 232.5 224)"/></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-client-left-top.svg b/2-ui/1-document/09-size-and-scroll/metric-client-left-top.svg index 968590466..f82bad607 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-client-left-top.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-client-left-top.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="353" height="316" viewBox="0 0 353 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top.svg"><path fill="#FFF" d="M0 0h353v316H0z"/><path id="Rectangle-206" fill="#DBAF88" d="M12 15h50v300H12z"/><path id="Rectangle-207" fill="#DBAF88" d="M177.5-115.5h50v311h-50z" transform="rotate(-90 202.5 40)"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="102" y="134">Introduction</tspan> <tspan x="102" y="183" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">This Ecma Standard is based on </tspan> <tspan x="102" y="216" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">several originating </tspan> <tspan x="102" y="249" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">technologies, the most well </tspan> <tspan x="102" y="282" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">known being JavaScript </tspan> <tspan x="102" y="315" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Netscape) and JScript </tspan> <tspan x="102" y="348" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Microsoft). The language was </tspan> <tspan x="102" y="381" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">invented by Brendan Eich at </tspan> <tspan x="102" y="414" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and first appeared in </tspan> <tspan x="102" y="447" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">that company’s Navigator 2.0 </tspan> <tspan x="102" y="480" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">browser. It has appeared in all </tspan> <tspan x="102" y="513" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">subsequent browsers from </tspan> <tspan x="102" y="546" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and in all browsers </tspan> <tspan x="102" y="579" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">from Microsoft starting with </tspan> <tspan x="102" y="612" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="102" y="645" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">The development of this </tspan> <tspan x="102" y="678" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Standard started in November </tspan> <tspan x="102" y="711" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">1996. The first edition of this </tspan> <tspan x="102" y="744" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Ecma Standard was adopted by </tspan> <tspan x="102" y="777" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">the Ecma General Assembly of </tspan> <tspan x="102" y="810" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">June 1997.</tspan></text><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M42-7h275v447H42z" transform="rotate(-180 179.5 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M316 8h162v399H316z"/><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M62.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H55.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M46.41 57.68l14 7-14 7-.001-6h-20l.001 6-14-7 14-7-.001 6h20l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="75.7" y="43" fill="#C06334">clientTop:</tspan> <tspan x="147.7" y="43" fill="#1C85B5">25px </tspan> <tspan x="183.7" y="43" fill="#C06334">= border</tspan></text><text id="clientLeft:25px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 37.32 136)"><tspan x="-16.68" y="140" fill="#C06334">clientLeft:</tspan> <tspan x="62.52" y="140" fill="#1C85B5">25px</tspan></text><g id="Rectangle-8-+-Rectangle-7" transform="translate(-45 -3)"><path id="Rectangle-8" fill="url(#linearGradient-1)" d="M86.5-85.5h275v447h-275z" transform="rotate(-90 224 138)"/><path id="Rectangle-7" fill="#FFF" d="M152 156h162v399H152z" transform="rotate(90 233 355.5)"/></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="353" height="316" viewBox="0 0 353 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top.svg"><path fill="#FFF" d="M0 0h353v316H0z"/><path id="Rectangle-206" fill="#DBAF88" d="M12 15h50v300H12z"/><path id="Rectangle-207" fill="#DBAF88" d="M177.5-115.5h50v311h-50z" transform="rotate(-90 202.5 40)"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="102" y="134">Introduction</tspan> <tspan x="102" y="183" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">This Ecma Standard is based on </tspan> <tspan x="102" y="216" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">several originating </tspan> <tspan x="102" y="249" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">technologies, the most well </tspan> <tspan x="102" y="282" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">known being JavaScript </tspan> <tspan x="102" y="315" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Netscape) and JScript </tspan> <tspan x="102" y="348" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Microsoft). The language was </tspan> <tspan x="102" y="381" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">invented by Brendan Eich at </tspan> <tspan x="102" y="414" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and first appeared in </tspan> <tspan x="102" y="447" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">that company’s Navigator 2.0 </tspan> <tspan x="102" y="480" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">browser. It has appeared in all </tspan> <tspan x="102" y="513" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">subsequent browsers from </tspan> <tspan x="102" y="546" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and in all browsers </tspan> <tspan x="102" y="579" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">from Microsoft starting with </tspan> <tspan x="102" y="612" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="102" y="645" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">The development of this </tspan> <tspan x="102" y="678" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Standard started in November </tspan> <tspan x="102" y="711" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">1996. The first edition of this </tspan> <tspan x="102" y="744" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Ecma Standard was adopted by </tspan> <tspan x="102" y="777" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">the Ecma General Assembly of </tspan> <tspan x="102" y="810" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">June 1997.</tspan></text><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M42-7h275v447H42z" transform="rotate(-180 179.5 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M316 8h162v399H316z"/><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M62.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H55.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M46.41 57.68l14 7-14 7-.001-6h-20l.001 6-14-7 14-7-.001 6h20l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="75.7" y="43" fill="#C06334">clientTop:</tspan> <tspan x="147.7" y="43" fill="#1C85B5">25px </tspan> <tspan x="183.7" y="43" fill="#C06334">= border</tspan></text><text id="clientLeft:25px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 37.32 136)"><tspan x="-16.68" y="140" fill="#C06334">clientLeft:</tspan> <tspan x="62.52" y="140" fill="#1C85B5">25px</tspan></text><g id="Rectangle-8-+-Rectangle-7" transform="translate(-45 -3)"><path id="Rectangle-8" fill="url(#linearGradient-1)" d="M86.5-85.5h275v447h-275z" transform="rotate(-90 224 138)"/><path id="Rectangle-7" fill="#FFF" d="M152 156h162v399H152z" transform="rotate(90 233 355.5)"/></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="353" height="316" viewBox="0 0 353 316"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="0%" x2="62.299%" y1="47.096%" y2="47.096%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#FFF" stop-opacity="0"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-left-top.svg"><path fill="#FFF" d="M0 0h353v316H0z"/><path id="Rectangle-206" fill="#DBAF88" d="M12 15h50v300H12z"/><path id="Rectangle-207" fill="#DBAF88" d="M177.5-115.5h50v311h-50z" transform="rotate(-90 202.5 40)"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="32" font-weight="bold"><tspan x="102" y="134">Introduction</tspan> <tspan x="102" y="183" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">This Ecma Standard is based on </tspan> <tspan x="102" y="216" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">several originating </tspan> <tspan x="102" y="249" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">technologies, the most well </tspan> <tspan x="102" y="282" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">known being JavaScript </tspan> <tspan x="102" y="315" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Netscape) and JScript </tspan> <tspan x="102" y="348" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">(Microsoft). The language was </tspan> <tspan x="102" y="381" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">invented by Brendan Eich at </tspan> <tspan x="102" y="414" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and first appeared in </tspan> <tspan x="102" y="447" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">that company’s Navigator 2.0 </tspan> <tspan x="102" y="480" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">browser. It has appeared in all </tspan> <tspan x="102" y="513" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">subsequent browsers from </tspan> <tspan x="102" y="546" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Netscape and in all browsers </tspan> <tspan x="102" y="579" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">from Microsoft starting with </tspan> <tspan x="102" y="612" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="102" y="645" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">The development of this </tspan> <tspan x="102" y="678" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Standard started in November </tspan> <tspan x="102" y="711" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">1996. The first edition of this </tspan> <tspan x="102" y="744" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">Ecma Standard was adopted by </tspan> <tspan x="102" y="777" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">the Ecma General Assembly of </tspan> <tspan x="102" y="810" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal">June 1997.</tspan></text><path id="Rectangle-5" fill="url(#linearGradient-1)" d="M42-7h275v447H42z" transform="rotate(-180 179.5 216.5)"/><path id="Rectangle-6" fill="#FFF" d="M316 8h162v399H316z"/><g id="Rectangle-8-+-Rectangle-7" transform="translate(-44.5 -2.5)"><path id="Rectangle-8" fill="url(#linearGradient-1)" d="M86-86h275v447H86z" transform="rotate(-90 223.5 137.5)"/><path id="Rectangle-7" fill="#FFF" d="M151.5 155.5h162v399h-162z" transform="rotate(90 232.5 355)"/></g><path id="Line-7" fill="#C06334" fill-rule="nonzero" d="M62.41 15.68l7 14h-6.001v20h6.001l-7 14-7-14h5.999v-20H55.41l7-14z"/><path id="Line-28" fill="#C06334" fill-rule="nonzero" d="M46.41 57.68l14 7-14 7-.001-6h-20l.001 6-14-7 14-7-.001 6h20l.001-6z"/><text id="clientTop:25px-=-bor" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="75.7" y="43" fill="#643B0C">clientTop:</tspan> <tspan x="147.7" y="43" fill="#166388">25px</tspan> <tspan x="176.5" y="43" fill="#643B0C"> = border</tspan></text><text id="clientLeft:25px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 37.32 136)"><tspan x="-16.68" y="140" fill="#643B0C">clientLeft:</tspan> <tspan x="62.52" y="140" fill="#166388">25px</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-client-width-height.svg b/2-ui/1-document/09-size-and-scroll/metric-client-width-height.svg index 83864b4c5..b5a342eab 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-client-width-height.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-client-width-height.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="500" height="493" viewBox="0 0 500 493"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M411 130v290H21V130h390zm-25 25H46v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M350 177v199H67V177h283z"/><g id="Group" transform="translate(370 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="12.9" y="75" fill="#C06334">border</tspan> <tspan x="20.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="32.3" y="45" fill="#C06334">padding</tspan> <tspan x="43.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="139.1" y="109" fill="#C06334">content width:</tspan> <tspan x="239.9" y="109" fill="#1C85B5">284px</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M336.5 110l14 7-14 7v-6H82.679l.001 6-14-7 14-7-.001 6H336.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M21.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="376.9" y="76" fill="#C06334">border</tspan> <tspan x="384.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="339.3" y="16" fill="#C06334">padding</tspan> <tspan x="350.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="349.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="367.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M385.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M410.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M66.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.48 93v41"/><text id="clientWidth-=-20+284" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="80.3" y="473" fill="#C06334">clientWidth = </tspan> <tspan x="197.9" y="473" fill="#1C85B5">20+284+20 </tspan> <tspan x="281.9" y="473" fill="#C06334">=</tspan> <tspan x="290.3" y="473" fill="#1C85B5"> 324px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M46.5 376v88.142"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M370.5 376v88.142"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M354 445.071l14 7-14 7v-6.001H62v6.001l-14-7 14-7v5.999h292v-5.999z"/><text id="clientHeight:240px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 488.5 277.5)"><tspan x="412.9" y="282" fill="#C06334">clientHeight:</tspan> <tspan x="522.1" y="282" fill="#1C85B5">240px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 156h120"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 394h120"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M476 160l7 14-6-.001V375h6l-7 14-7-14h6V173.999l-6 .001 7-14z"/><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 422.5 274)"><tspan x="379.3" y="278" fill="#C06334">height:</tspan> <tspan x="429.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M435 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="69" y="193">Introduction</tspan> <tspan x="69" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="69" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="69" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="69" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="69" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="69" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="69" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="69" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="69" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="500" height="493" viewBox="0 0 500 493"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M411 130v290H21V130h390zm-25 25H46v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M350 177v199H67V177h283z"/><g id="Group" transform="translate(370 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="12.9" y="75" fill="#C06334">border</tspan> <tspan x="20.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="32.3" y="45" fill="#C06334">padding</tspan> <tspan x="43.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="139.1" y="109" fill="#C06334">content width:</tspan> <tspan x="239.9" y="109" fill="#1C85B5">284px</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M336.5 110l14 7-14 7v-6H82.679l.001 6-14-7 14-7-.001 6H336.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M21.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="376.9" y="76" fill="#C06334">border</tspan> <tspan x="384.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="339.3" y="16" fill="#C06334">padding</tspan> <tspan x="350.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="349.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="367.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M385.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M410.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M66.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.48 93v41"/><text id="clientWidth-=-20+284" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="80.3" y="473" fill="#C06334">clientWidth = </tspan> <tspan x="197.9" y="473" fill="#1C85B5">20+284+20 </tspan> <tspan x="281.9" y="473" fill="#C06334">=</tspan> <tspan x="290.3" y="473" fill="#1C85B5"> 324px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M46.5 376v88.142"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M370.5 376v88.142"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M354 445.071l14 7-14 7v-6.001H62v6.001l-14-7 14-7v5.999h292v-5.999z"/><text id="clientHeight:240px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 488.5 277.5)"><tspan x="412.9" y="282" fill="#C06334">clientHeight:</tspan> <tspan x="522.1" y="282" fill="#1C85B5">240px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 156h120"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 394h120"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M476 160l7 14-6-.001V375h6l-7 14-7-14h6V173.999l-6 .001 7-14z"/><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 422.5 274)"><tspan x="379.3" y="278" fill="#C06334">height:</tspan> <tspan x="429.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M435 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="69" y="193">Introduction</tspan> <tspan x="69" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="69" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="69" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="69" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="69" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="69" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="69" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="69" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="69" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="500" height="493" viewBox="0 0 500 493"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-client-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M411 130v290H21V130h390zm-25 25H46v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M350 177v199H67V177h283z"/><g id="Group-2" transform="translate(370 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="12.9" y="75" fill="#C06334">border</tspan> <tspan x="20.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="32.3" y="45" fill="#C06334">padding</tspan> <tspan x="43.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="139.1" y="109" fill="#C06334">content width:</tspan> <tspan x="239.9" y="109" fill="#1C85B5">284px</tspan></text><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M336.5 110l14 7-14 7v-6H82.679l.001 6-14-7 14-7-.001 6H336.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M21.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="376.9" y="76" fill="#C06334">border</tspan> <tspan x="384.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="339.3" y="16" fill="#C06334">padding</tspan> <tspan x="350.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="349.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="367.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M385.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M410.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M66.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.48 93v41"/><text id="clientWidth-=-20+284" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="80.3" y="473" fill="#C06334">clientWidth = </tspan> <tspan x="197.9" y="473" fill="#1C85B5">20+284+20 </tspan> <tspan x="281.9" y="473" fill="#C06334">=</tspan> <tspan x="290.3" y="473" fill="#1C85B5"> 324px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M46.5 376v88.142"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M370.5 376v88.142"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M354 445.071l14 7-14 7v-6.001H62v6.001l-14-7 14-7v5.999h292v-5.999z"/><text id="clientHeight:240px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 488.5 277.5)"><tspan x="412.9" y="282" fill="#C06334">clientHeight:</tspan> <tspan x="522.1" y="282" fill="#1C85B5">240px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 156h120"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M365.5 394h120"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M476 160l7 14-6-.001V375h6l-7 14-7-14h6V173.999l-6 .001 7-14z"/><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 422.5 274)"><tspan x="379.3" y="278" fill="#C06334">height:</tspan> <tspan x="429.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M353.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M435 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="69" y="193">Introduction</tspan> <tspan x="69" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="69" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="69" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="69" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="69" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="69" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="69" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="69" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="69" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-css.svg b/2-ui/1-document/09-size-and-scroll/metric-css.svg index 13aa62afd..3d2fb3781 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-css.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-css.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="566" height="469" viewBox="0 0 566 469"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-css.svg"><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M460 184v199H177V184h283z"/><path id="Rectangle-1" fill="#DBAF88" d="M521 139v290H131V139h390zm-25 25H156v240h340V164z"/><g id="Group" transform="translate(480 163)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="padding:20px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="176" fill="#C06334">padding:</tspan> <tspan x="64.6" y="176" fill="#1C85B5">20px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 530.5 282)"><tspan x="487.3" y="286" fill="#C06334">height:</tspan> <tspan x="537.7" y="286" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 183h88.142"/><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 184h88.142"/><path id="Line-4" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 162h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 385h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M543 185.5l7 14h-6V368h6l-7 14-7-14h6V199.5h-6l7-14z"/><text id="padding:20px-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="396" fill="#C06334">padding:</tspan> <tspan x="64.6" y="396" fill="#1C85B5">20px</tspan></text><path id="Line-5" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 404h88.142"/><path id="Line-6" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 382h88.142"/><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="122.9" y="83" fill="#C06334">border</tspan> <tspan x="130.1" y="97" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="142.3" y="53" fill="#C06334">padding</tspan> <tspan x="153.1" y="67" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="249.1" y="117" fill="#C06334">content width:</tspan> <tspan x="349.9" y="117" fill="#1C85B5">284px</tspan></text><path id="Line-21" fill="#C06334" fill-rule="nonzero" d="M446.5 118l14 7-14 7v-6H192.679l.001 6-14-7 14-7-.001 6H446.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M131.48 101v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M157.48 101v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="486.9" y="84" fill="#C06334">border</tspan> <tspan x="494.1" y="98" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="449.3" y="24" fill="#C06334">padding</tspan> <tspan x="460.1" y="38" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="459.1" y="56" fill="#C06334">scrollbar</tspan> <tspan x="477.1" y="70" fill="#1C85B5">16px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="178" y="200">Introduction</tspan> <tspan x="178" y="228" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="178" y="247" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="178" y="266" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="178" y="285" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="178" y="304" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="178" y="323" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="178" y="342" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="178" y="361" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="178" y="380" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with</tspan> <tspan x="178" y="399" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> </tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M481.48 101v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M495.48 101v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M520.48 101v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M176.48 101v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M463.48 101v41"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="566" height="469" viewBox="0 0 566 469"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-css.svg"><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M460 184v199H177V184h283z"/><path id="Rectangle-1" fill="#DBAF88" d="M521 139v290H131V139h390zm-25 25H156v240h340V164z"/><g id="Group" transform="translate(480 163)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="padding:20px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="176" fill="#C06334">padding:</tspan> <tspan x="64.6" y="176" fill="#1C85B5">20px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 530.5 282)"><tspan x="487.3" y="286" fill="#C06334">height:</tspan> <tspan x="537.7" y="286" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 183h88.142"/><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 184h88.142"/><path id="Line-4" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 162h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 385h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M543 185.5l7 14h-6V368h6l-7 14-7-14h6V199.5h-6l7-14z"/><text id="padding:20px-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="396" fill="#C06334">padding:</tspan> <tspan x="64.6" y="396" fill="#1C85B5">20px</tspan></text><path id="Line-5" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 404h88.142"/><path id="Line-6" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 382h88.142"/><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="122.9" y="83" fill="#C06334">border</tspan> <tspan x="130.1" y="97" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="142.3" y="53" fill="#C06334">padding</tspan> <tspan x="153.1" y="67" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="249.1" y="117" fill="#C06334">content width:</tspan> <tspan x="349.9" y="117" fill="#1C85B5">284px</tspan></text><path id="Line-21" fill="#C06334" fill-rule="nonzero" d="M446.5 118l14 7-14 7v-6H192.679l.001 6-14-7 14-7-.001 6H446.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M131.48 101v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M157.48 101v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="486.9" y="84" fill="#C06334">border</tspan> <tspan x="494.1" y="98" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="449.3" y="24" fill="#C06334">padding</tspan> <tspan x="460.1" y="38" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="459.1" y="56" fill="#C06334">scrollbar</tspan> <tspan x="477.1" y="70" fill="#1C85B5">16px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="178" y="200">Introduction</tspan> <tspan x="178" y="228" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="178" y="247" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="178" y="266" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="178" y="285" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="178" y="304" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="178" y="323" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="178" y="342" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="178" y="361" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="178" y="380" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with</tspan> <tspan x="178" y="399" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> </tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M481.48 101v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M495.48 101v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M520.48 101v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M176.48 101v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M463.48 101v41"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="566" height="469" viewBox="0 0 566 469"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-css.svg"><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M460 184v199H177V184h283z"/><path id="Rectangle-1" fill="#DBAF88" d="M521 139v290H131V139h390zm-25 25H156v240h340V164z"/><g id="Group-2" transform="translate(480 163)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="padding:20px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="176" fill="#C06334">padding:</tspan> <tspan x="64.6" y="176" fill="#1C85B5">20px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 530.5 282)"><tspan x="487.3" y="286" fill="#C06334">height:</tspan> <tspan x="537.7" y="286" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 183h88.142"/><path id="Line-3" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 184h88.142"/><path id="Line-4" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 162h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M461.5 385h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M543 185.5l7 14h-6V368h6l-7 14-7-14h6V199.5h-6l7-14z"/><text id="padding:20px-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="7" y="396" fill="#C06334">padding:</tspan> <tspan x="64.6" y="396" fill="#1C85B5">20px</tspan></text><path id="Line-5" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 404h88.142"/><path id="Line-6" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M93.5 382h88.142"/><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="122.9" y="83" fill="#C06334">border</tspan> <tspan x="130.1" y="97" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="142.3" y="53" fill="#C06334">padding</tspan> <tspan x="153.1" y="67" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="249.1" y="117" fill="#C06334">content width:</tspan> <tspan x="349.9" y="117" fill="#1C85B5">284px</tspan></text><path id="Line-21" fill="#C06334" fill-rule="nonzero" d="M446.5 118l14 7-14 7v-6H192.679l.001 6-14-7 14-7-.001 6H446.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M131.48 101v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M157.48 101v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="486.9" y="84" fill="#C06334">border</tspan> <tspan x="494.1" y="98" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="449.3" y="24" fill="#C06334">padding</tspan> <tspan x="460.1" y="38" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="459.1" y="56" fill="#C06334">scrollbar</tspan> <tspan x="477.1" y="70" fill="#1C85B5">16px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="178" y="200">Introduction</tspan> <tspan x="178" y="228" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="178" y="247" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="178" y="266" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="178" y="285" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="178" y="304" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="178" y="323" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="178" y="342" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="178" y="361" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="178" y="380" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with</tspan> <tspan x="178" y="399" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> </tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M481.48 101v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M495.48 101v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M520.48 101v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M176.48 101v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M463.48 101v41"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-offset-parent.svg b/2-ui/1-document/09-size-and-scroll/metric-offset-parent.svg index 9e247639b..563b7e150 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-offset-parent.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-offset-parent.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="597" height="520" viewBox="0 0 597 520"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-4" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><pattern id="pattern-1" width="30" height="30" x="-12" y="-6" patternUnits="userSpaceOnUse"><use xlink:href="#image-2"/></pattern><image id="image-2" width="30" height="30" xlink:href=""/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-parent.svg"><path fill="#FFF" d="M0 0h597v520H0z"/><path id="Rectangle-10" fill="url(#pattern-1)" fill-opacity=".5" stroke="#D1CFCD" stroke-width="3" d="M19.5 25.5h558v479h-558z"/><path id="Rectangle-1" fill="#DBAF88" d="M552 185v290H178V185h374zm-25 25H203v240h324V210z"/><text id="offsetTop:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 163 104.32)"><tspan x="100" y="108.82" fill="#C06334">offsetTop:</tspan> <tspan x="184" y="108.82" fill="#1C85B5">180px</tspan></text><text id="offsetLeft:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="34.3" y="179.32" fill="#C06334">offsetLeft:</tspan> <tspan x="126.7" y="179.32" fill="#1C85B5">180px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="223" y="247">Introduction</tspan> <tspan x="223" y="275" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="223" y="294" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="223" y="313" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="223" y="332" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="223" y="351" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="223" y="370" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="223" y="389" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="223" y="408" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="223" y="427" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="223" y="446" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M179 26.5l7 14h-6v128.82h6l-7 14-7-14h6V40.5h-6l7-14z"/><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M163.41 179.91l14 7-14 7v-6H34.589l.001 6-14-7 14-7-.001 6H163.41v-6z"/><circle id="Oval-2" cx="203" cy="211" r="2" fill="#1C85B5"/><circle id="Oval-2" cx="179" cy="186" r="3" fill="#1C85B5"/><ellipse id="Oval-6" cx="364.25" cy="104" fill="#FBF2EC" stroke="#C06334" stroke-width="2" rx="100" ry="50"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M366 156.379v11.638h6l-7 14-7-14h6v-11.638h2z"/><text id="position:-absolute;" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="296.5" y="95.793">position: absolute;</tspan> <tspan x="296.5" y="110.793">left: 180px;</tspan> <tspan x="296.5" y="125.793">top: 180px;</tspan></text><text id="offsetParent-<MAIN>-" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="20.5" y="19.379">offsetParent <MAIN> </tspan></text><text id="<DIV>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="183.5" y="180.379"><DIV></tspan></text><g id="Group" transform="translate(511 210)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-4)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="597" height="520" viewBox="0 0 597 520"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-4" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><pattern id="pattern-1" width="30" height="30" x="-12" y="-6" patternUnits="userSpaceOnUse"><use xlink:href="#image-2"/></pattern><image id="image-2" width="30" height="30" xlink:href=""/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-parent.svg"><path fill="#FFF" d="M0 0h597v520H0z"/><path id="Rectangle-10" fill="url(#pattern-1)" fill-opacity=".5" stroke="#D1CFCD" stroke-width="3" d="M19.5 25.5h558v479h-558z"/><path id="Rectangle-1" fill="#DBAF88" d="M552 185v290H178V185h374zm-25 25H203v240h324V210z"/><text id="offsetTop:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 163 104.32)"><tspan x="100" y="108.82" fill="#C06334">offsetTop:</tspan> <tspan x="184" y="108.82" fill="#1C85B5">180px</tspan></text><text id="offsetLeft:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="34.3" y="179.32" fill="#C06334">offsetLeft:</tspan> <tspan x="126.7" y="179.32" fill="#1C85B5">180px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="223" y="247">Introduction</tspan> <tspan x="223" y="275" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="223" y="294" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="223" y="313" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="223" y="332" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="223" y="351" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="223" y="370" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="223" y="389" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="223" y="408" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="223" y="427" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="223" y="446" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M179 26.5l7 14h-6v128.82h6l-7 14-7-14h6V40.5h-6l7-14z"/><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M163.41 179.91l14 7-14 7v-6H34.589l.001 6-14-7 14-7-.001 6H163.41v-6z"/><circle id="Oval-2" cx="203" cy="211" r="2" fill="#1C85B5"/><circle id="Oval-2" cx="179" cy="186" r="3" fill="#1C85B5"/><ellipse id="Oval-6" cx="364.25" cy="104" fill="#FBF2EC" stroke="#C06334" stroke-width="2" rx="100" ry="50"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M366 156.379v11.638h6l-7 14-7-14h6v-11.638h2z"/><text id="position:-absolute;" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="296.5" y="95.793">position: absolute;</tspan> <tspan x="296.5" y="110.793">left: 180px;</tspan> <tspan x="296.5" y="125.793">top: 180px;</tspan></text><text id="offsetParent-<MAIN>-" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="20.5" y="19.379">offsetParent <MAIN> </tspan></text><text id="<DIV>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="183.5" y="180.379"><DIV></tspan></text><g id="Group" transform="translate(511 210)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-4)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="597" height="520" viewBox="0 0 597 520"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-3" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-4" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><pattern id="pattern-1" width="30" height="30" x="-12" y="-6" patternUnits="userSpaceOnUse"><use xlink:href="#image-2"/></pattern><image id="image-2" width="30" height="30" xlink:href=""/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-parent.svg"><path fill="#FFF" d="M0 0h597v520H0z"/><path id="Rectangle-10" fill="url(#pattern-1)" fill-opacity=".5" stroke="#D1CFCD" stroke-width="3" d="M19.5 25.5h558v479h-558z"/><path id="Rectangle-1" fill="#DBAF88" d="M552 185v290H178V185h374zm-25 25H203v240h324V210z"/><text id="offsetTop:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 163 104.32)"><tspan x="100" y="108.82" fill="#C06334">offsetTop:</tspan> <tspan x="184" y="108.82" fill="#1C85B5">180px</tspan></text><text id="offsetLeft:180px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="34.3" y="179.32" fill="#C06334">offsetLeft:</tspan> <tspan x="126.7" y="179.32" fill="#1C85B5">180px</tspan></text><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="223" y="247">Introduction</tspan> <tspan x="223" y="275" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="223" y="294" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="223" y="313" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="223" y="332" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="223" y="351" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="223" y="370" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="223" y="389" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="223" y="408" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="223" y="427" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="223" y="446" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft</tspan></text><path id="Line-40" fill="#C06334" fill-rule="nonzero" d="M179 26.5l7 14h-6v128.82h6l-7 14-7-14h6V40.5h-6l7-14z"/><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M163.41 179.91l14 7-14 7v-6H34.589l.001 6-14-7 14-7-.001 6H163.41v-6z"/><circle id="Oval-2" cx="203" cy="211" r="2" fill="#1C85B5"/><circle id="Oval-2" cx="179" cy="186" r="3" fill="#1C85B5"/><ellipse id="Oval-6" cx="364.25" cy="104" fill="#FBF2EC" stroke="#C06334" stroke-width="2" rx="100" ry="50"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M366 156.379v11.638h6l-7 14-7-14h6v-11.638h2z"/><text id="position:-absolute;" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="296.5" y="95.793">position: absolute;</tspan> <tspan x="296.5" y="110.793">left: 180px;</tspan> <tspan x="296.5" y="125.793">top: 180px;</tspan></text><text id="offsetParent-<MAIN>-" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="20.5" y="19.379">offsetParent <MAIN> </tspan></text><text id="<DIV>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="183.5" y="180.379"><DIV></tspan></text><g id="Scrollbar" transform="translate(511 210)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-3)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-4)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-offset-width-height.svg b/2-ui/1-document/09-size-and-scroll/metric-offset-width-height.svg index 49bdccda7..c7fe3ac1d 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-offset-width-height.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-offset-width-height.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="508" height="509" viewBox="0 0 508 509"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M420 130v290H30V130h390zm-25 25H55v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M359 177v199H76V177h283z"/><g id="Group" transform="translate(379 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="21.9" y="75" fill="#C06334">border</tspan> <tspan x="29.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="41.3" y="45" fill="#C06334">padding</tspan> <tspan x="52.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="148.1" y="109" fill="#C06334">content width:</tspan> <tspan x="248.9" y="109" fill="#1C85B5">284px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 429.5 274)"><tspan x="386.3" y="278" fill="#C06334">height:</tspan> <tspan x="436.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M442 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M345.5 110l14 7-14 7v-6H91.679l.001 6-14-7 14-7-.001 6H345.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M56.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="385.9" y="76" fill="#C06334">border</tspan> <tspan x="393.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.3" y="16" fill="#C06334">padding</tspan> <tspan x="359.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="358.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="376.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M380.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M394.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M75.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M362.48 93v41"/><text id="offsetWidth-=-25+20+" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="56.5" y="484" fill="#C06334">offsetWidth = </tspan> <tspan x="174.1" y="484" fill="#1C85B5">25+20+284+20+16+25 </tspan> <tspan x="333.7" y="484" fill="#C06334">=</tspan> <tspan x="342.1" y="484" fill="#1C85B5"> 390px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.5 419v78"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.5 419v80"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M403 456.071l14 7-14 7v-6.001H47.089l.001 6.001-14-7 14-7-.001 5.999H403v-5.999z"/><text id="offsetHeight:290px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 483.5 280.5)"><tspan x="407.9" y="285" fill="#C06334">offsetHeight:</tspan> <tspan x="517.1" y="285" fill="#1C85B5">290px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 131h88.142"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 420h88.142"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M469 132.59l7 14-6-.001V405.41h6l-7 14-7-14h6V146.589l-6 .001 7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="79" y="193">Introduction</tspan> <tspan x="79" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="79" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="79" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="79" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="79" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="79" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="79" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="79" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="79" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting </tspan> <tspan x="79" y="392" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">with </tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="508" height="509" viewBox="0 0 508 509"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M420 130v290H30V130h390zm-25 25H55v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M359 177v199H76V177h283z"/><g id="Group" transform="translate(379 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="21.9" y="75" fill="#C06334">border</tspan> <tspan x="29.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="41.3" y="45" fill="#C06334">padding</tspan> <tspan x="52.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="148.1" y="109" fill="#C06334">content width:</tspan> <tspan x="248.9" y="109" fill="#1C85B5">284px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 429.5 274)"><tspan x="386.3" y="278" fill="#C06334">height:</tspan> <tspan x="436.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M442 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M345.5 110l14 7-14 7v-6H91.679l.001 6-14-7 14-7-.001 6H345.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M56.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="385.9" y="76" fill="#C06334">border</tspan> <tspan x="393.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.3" y="16" fill="#C06334">padding</tspan> <tspan x="359.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="358.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="376.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M380.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M394.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M75.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M362.48 93v41"/><text id="offsetWidth-=-25+20+" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="56.5" y="484" fill="#C06334">offsetWidth = </tspan> <tspan x="174.1" y="484" fill="#1C85B5">25+20+284+20+16+25 </tspan> <tspan x="333.7" y="484" fill="#C06334">=</tspan> <tspan x="342.1" y="484" fill="#1C85B5"> 390px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.5 419v78"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.5 419v80"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M403 456.071l14 7-14 7v-6.001H47.089l.001 6.001-14-7 14-7-.001 5.999H403v-5.999z"/><text id="offsetHeight:290px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 483.5 280.5)"><tspan x="407.9" y="285" fill="#C06334">offsetHeight:</tspan> <tspan x="517.1" y="285" fill="#1C85B5">290px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 131h88.142"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 420h88.142"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M469 132.59l7 14-6-.001V405.41h6l-7 14-7-14h6V146.589l-6 .001 7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="79" y="193">Introduction</tspan> <tspan x="79" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="79" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="79" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="79" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="79" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="79" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="79" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="79" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="79" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting </tspan> <tspan x="79" y="392" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">with </tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="508" height="509" viewBox="0 0 508 509"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-offset-width-height.svg"><path id="Rectangle-2" fill="#DBAF88" d="M420 130v290H30V130h390zm-25 25H55v240h340V155z"/><path id="Rectangle-1" stroke="#C06334" stroke-width="2" d="M359 177v199H76V177h283z"/><g id="scrollbar" transform="translate(379 155)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g><text id="border" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="21.9" y="75" fill="#C06334">border</tspan> <tspan x="29.1" y="89" fill="#1C85B5">25px</tspan></text><text id="padding" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="41.3" y="45" fill="#C06334">padding</tspan> <tspan x="52.1" y="59" fill="#1C85B5">20px</tspan></text><text id="content-width:284px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="148.1" y="109" fill="#C06334">content width:</tspan> <tspan x="248.9" y="109" fill="#1C85B5">284px</tspan></text><text id="height:200px" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 429.5 274)"><tspan x="386.3" y="278" fill="#C06334">height:</tspan> <tspan x="436.7" y="278" fill="#1C85B5">200px</tspan></text><path id="Line" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 175h88.142"/><path id="Line-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M360.5 377h88.142"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M442 177.5l7 14h-6V360h6l-7 14-7-14h6V191.5h-6l7-14z"/><path id="Line-15" fill="#C06334" fill-rule="nonzero" d="M345.5 110l14 7-14 7v-6H91.679l.001 6-14-7 14-7-.001 6H345.5v-6z"/><path id="Line-14" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.48 93v43"/><path id="Line-13" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M56.48 93v41"/><text id="border-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="385.9" y="76" fill="#C06334">border</tspan> <tspan x="393.1" y="90" fill="#1C85B5">25px</tspan></text><text id="padding-2" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="348.3" y="16" fill="#C06334">padding</tspan> <tspan x="359.1" y="30" fill="#1C85B5">20px</tspan></text><text id="scrollbar" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="358.1" y="48" fill="#C06334">scrollbar</tspan> <tspan x="376.1" y="62" fill="#1C85B5">16px</tspan></text><path id="Line-17" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M380.48 93v43"/><path id="Line-20" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M394.48 93v43"/><path id="Line-18" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.48 93v41"/><path id="Line-16" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M75.48 93v41"/><path id="Line-19" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M362.48 93v41"/><text id="offsetWidth-=-25+20+" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="56.5" y="484" fill="#C06334">offsetWidth = </tspan> <tspan x="174.1" y="484" fill="#1C85B5">25+20+284+20+16+25 </tspan> <tspan x="333.7" y="484" fill="#C06334">=</tspan> <tspan x="342.1" y="484" fill="#1C85B5"> 390px</tspan></text><path id="Line-24" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M30.5 419v78"/><path id="Line-25" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M419.5 419v80"/><path id="Line-22" fill="#C06334" fill-rule="nonzero" d="M403 456.071l14 7-14 7v-6.001H47.089l.001 6.001-14-7 14-7-.001 5.999H403v-5.999z"/><text id="offsetHeight:290px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 483.5 280.5)"><tspan x="407.9" y="285" fill="#C06334">offsetHeight:</tspan> <tspan x="517.1" y="285" fill="#1C85B5">290px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 131h88.142"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M417.5 420h88.142"/><path id="Line-26" fill="#C06334" fill-rule="nonzero" d="M469 132.59l7 14-6-.001V405.41h6l-7 14-7-14h6V146.589l-6 .001 7-14z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="79" y="193">Introduction</tspan> <tspan x="79" y="221" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="79" y="240" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="79" y="259" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="79" y="278" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="79" y="297" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="79" y="316" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="79" y="335" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="79" y="354" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="79" y="373" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting </tspan> <tspan x="79" y="392" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">with </tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-scroll-top.svg b/2-ui/1-document/09-size-and-scroll/metric-scroll-top.svg index c6d14d0f3..d3dbcf9c8 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-scroll-top.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-scroll-top.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="489" height="542" viewBox="0 0 489 542"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-top.svg"><path fill="#FFF" d="M0 0h489v542H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="94" y="54">Introduction</tspan> <tspan x="94" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="94" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="94" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="94" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="94" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="94" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="94" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="94" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="94" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="94" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="94" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="94" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="94" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="94" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="94" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="94" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="94" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="94" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="94" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="94" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="94" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="94" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="94" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M425 122v290H51V122h374zm-25 25H76v240h324V147z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M75 22h326v500H75z"/><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 16.5 84)"><tspan x="-15.9" y="88">scrollTop</tspan></text><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M9.5 147h66.14"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M10.5 21h66.14"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M35 20.5l7 14h-6v97.819l6 .001-7 14-7-14 6-.001V34.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M88 33h312v89H88z"/><path id="Rectangle-15" fill="#FFF" d="M88 411h312v89H88z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 449 270)"><tspan x="373.4" y="274.5" fill="#C06334">scrollHeight:</tspan> <tspan x="482.6" y="274.5" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 22h78.14"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 522h78.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M462 24.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><g id="Group" transform="translate(384 147)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="489" height="542" viewBox="0 0 489 542"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-top.svg"><path fill="#FFF" d="M0 0h489v542H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="94" y="54">Introduction</tspan> <tspan x="94" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="94" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="94" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="94" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="94" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="94" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="94" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="94" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="94" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="94" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="94" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="94" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="94" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="94" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="94" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="94" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="94" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="94" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="94" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="94" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="94" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="94" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="94" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M425 122v290H51V122h374zm-25 25H76v240h324V147z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M75 22h326v500H75z"/><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 16.5 84)"><tspan x="-15.9" y="88">scrollTop</tspan></text><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M9.5 147h66.14"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M10.5 21h66.14"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M35 20.5l7 14h-6v97.819l6 .001-7 14-7-14 6-.001V34.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M88 33h312v89H88z"/><path id="Rectangle-15" fill="#FFF" d="M88 411h312v89H88z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 449 270)"><tspan x="373.4" y="274.5" fill="#C06334">scrollHeight:</tspan> <tspan x="482.6" y="274.5" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 22h78.14"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 522h78.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M462 24.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><g id="Group" transform="translate(384 147)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="489" height="542" viewBox="0 0 489 542"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-top.svg"><path fill="#FFF" d="M0 0h489v542H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="94" y="54">Introduction</tspan> <tspan x="94" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="94" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="94" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="94" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="94" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="94" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="94" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. It has appeared in all </tspan> <tspan x="94" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">subsequent browsers from Netscape and </tspan> <tspan x="94" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all browsers from Microsoft starting with </tspan> <tspan x="94" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Internet Explorer 3.0.</tspan> <tspan x="94" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard started </tspan> <tspan x="94" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in November 1996. The first edition of this </tspan> <tspan x="94" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma Standard was adopted by the Ecma </tspan> <tspan x="94" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">General Assembly of June 1997.</tspan> <tspan x="94" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="94" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="94" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="94" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="94" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="94" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="94" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="94" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="94" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-1" fill="#DBAF88" d="M425 122v290H51V122h374zm-25 25H76v240h324V147z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M75 22h326v500H75z"/><text id="scrollTop" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal" transform="rotate(-90 16.5 84)"><tspan x="-15.9" y="88">scrollTop</tspan></text><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M9.5 147h66.14"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M10.5 21h66.14"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M35 20.5l7 14h-6v97.819l6 .001-7 14-7-14 6-.001V34.5h-6l7-14z"/><path id="Rectangle-14" fill="#FFF" d="M88 33h312v89H88z"/><path id="Rectangle-15" fill="#FFF" d="M88 411h312v89H88z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 449 270)"><tspan x="373.4" y="274.5" fill="#C06334">scrollHeight:</tspan> <tspan x="482.6" y="274.5" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 22h78.14"/><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M393.5 522h78.14"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M462 24.5l7 14h-6v466h6l-7 14-7-14h6v-466h-6l7-14z"/><g id="Scrollbar" transform="translate(384 147)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/09-size-and-scroll/metric-scroll-width-height.svg b/2-ui/1-document/09-size-and-scroll/metric-scroll-width-height.svg index 0c3d29952..9865fd8d5 100644 --- a/2-ui/1-document/09-size-and-scroll/metric-scroll-width-height.svg +++ b/2-ui/1-document/09-size-and-scroll/metric-scroll-width-height.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="463" height="524" viewBox="0 0 463 524"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-width-height.svg"><path fill="#FFF" d="M0 0h463v524H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="66" y="54">Introduction</tspan> <tspan x="66" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="66" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="66" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="66" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="66" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="66" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="66" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="66" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="66" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="66" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft starting with Internet Explorer </tspan> <tspan x="66" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">3.0.</tspan> <tspan x="66" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard </tspan> <tspan x="66" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">started in November 1996. The first </tspan> <tspan x="66" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of this Ecma Standard was </tspan> <tspan x="66" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">adopted by the Ecma General Assembly </tspan> <tspan x="66" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1997.</tspan> <tspan x="66" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to </tspan> <tspan x="66" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">ISO/IEC JTC 1 for adoption under the </tspan> <tspan x="66" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">fast-track procedure, and approved as </tspan> <tspan x="66" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">international standard ISO/IEC 16262, in </tspan> <tspan x="66" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">April 1998. The Ecma General Assembly </tspan> <tspan x="66" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1998 approved the second </tspan> <tspan x="66" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of ECMA-262 to keep it fully </tspan> <tspan x="66" y="519" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">aligned with ISO/IEC 16262. Changes </tspan> <tspan x="66" y="538" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">between the first and the second </tspan> <tspan x="66" y="557" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-15" fill="#FFF" d="M58 410h312v111H58z"/><path id="Rectangle-14" fill="#FFF" d="M58 35h312v89H58z"/><path id="Rectangle-1" fill="#DBAF88" d="M395 123v290H21V123h374zm-25 25H46v240h324V148z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M45 22h326v502H45z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 426 270.5)"><tspan x="350.4" y="275" fill="#C06334">scrollHeight:</tspan> <tspan x="459.6" y="275" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 22h78"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 524h78"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M439 26l7 14h-6v466h6l-7 14-7-14h6V40h-6l7-14z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M335.36 102l14 7-14 7-.001-6H64v6l-14-7 14-7v6h271.359l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M352 154v-54"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47 154v-54"/><text id="scrollWidth-=-324px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="149.2" y="105" fill="#C06334">scrollWidth = </tspan> <tspan x="266.8" y="105" fill="#1C85B5">324px</tspan></text><g id="Group" transform="translate(354 148)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="524" viewBox="0 0 463 524"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-width-height.svg"><path fill="#FFF" d="M0 0h463v524H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="66" y="54">Introduction</tspan> <tspan x="66" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="66" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="66" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="66" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="66" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="66" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="66" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="66" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="66" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="66" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft starting with Internet Explorer </tspan> <tspan x="66" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">3.0.</tspan> <tspan x="66" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard </tspan> <tspan x="66" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">started in November 1996. The first </tspan> <tspan x="66" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of this Ecma Standard was </tspan> <tspan x="66" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">adopted by the Ecma General Assembly </tspan> <tspan x="66" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1997.</tspan> <tspan x="66" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to </tspan> <tspan x="66" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">ISO/IEC JTC 1 for adoption under the </tspan> <tspan x="66" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">fast-track procedure, and approved as </tspan> <tspan x="66" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">international standard ISO/IEC 16262, in </tspan> <tspan x="66" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">April 1998. The Ecma General Assembly </tspan> <tspan x="66" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1998 approved the second </tspan> <tspan x="66" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of ECMA-262 to keep it fully </tspan> <tspan x="66" y="519" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">aligned with ISO/IEC 16262. Changes </tspan> <tspan x="66" y="538" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">between the first and the second </tspan> <tspan x="66" y="557" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-15" fill="#FFF" d="M58 410h312v111H58z"/><path id="Rectangle-14" fill="#FFF" d="M58 35h312v89H58z"/><path id="Rectangle-1" fill="#DBAF88" d="M395 123v290H21V123h374zm-25 25H46v240h324V148z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M45 22h326v502H45z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 426 270.5)"><tspan x="350.4" y="275" fill="#C06334">scrollHeight:</tspan> <tspan x="459.6" y="275" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 22h78"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 524h78"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M439 26l7 14h-6v466h6l-7 14-7-14h6V40h-6l7-14z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M335.36 102l14 7-14 7-.001-6H64v6l-14-7 14-7v6h271.359l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M352 154v-54"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47 154v-54"/><text id="scrollWidth-=-324px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="149.2" y="105" fill="#C06334">scrollWidth = </tspan> <tspan x="266.8" y="105" fill="#1C85B5">324px</tspan></text><g id="Group" transform="translate(354 148)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="524" viewBox="0 0 463 524"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><linearGradient id="linearGradient-1" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient><linearGradient id="linearGradient-2" x1="50%" x2="50%" y1="0%" y2="100%"><stop offset="0%" stop-color="#FFF"/><stop offset="100%" stop-color="#D1CFCD"/></linearGradient></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="metric-scroll-width-height.svg"><path fill="#FFF" d="M0 0h463v524H0z"/><text id="Introduction" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="66" y="54">Introduction</tspan> <tspan x="66" y="82" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="66" y="101" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="66" y="120" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="66" y="139" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="66" y="158" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape </tspan> <tspan x="66" y="177" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">and first appeared in that company’s </tspan> <tspan x="66" y="196" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Navigator 2.0 browser. It has appeared </tspan> <tspan x="66" y="215" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">in all subsequent browsers from </tspan> <tspan x="66" y="234" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Netscape and in all browsers from </tspan> <tspan x="66" y="253" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft starting with Internet Explorer </tspan> <tspan x="66" y="272" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">3.0.</tspan> <tspan x="66" y="291" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The development of this Standard </tspan> <tspan x="66" y="310" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">started in November 1996. The first </tspan> <tspan x="66" y="329" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of this Ecma Standard was </tspan> <tspan x="66" y="348" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">adopted by the Ecma General Assembly </tspan> <tspan x="66" y="367" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1997.</tspan> <tspan x="66" y="386" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to </tspan> <tspan x="66" y="405" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">ISO/IEC JTC 1 for adoption under the </tspan> <tspan x="66" y="424" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">fast-track procedure, and approved as </tspan> <tspan x="66" y="443" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">international standard ISO/IEC 16262, in </tspan> <tspan x="66" y="462" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">April 1998. The Ecma General Assembly </tspan> <tspan x="66" y="481" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">of June 1998 approved the second </tspan> <tspan x="66" y="500" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition of ECMA-262 to keep it fully </tspan> <tspan x="66" y="519" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">aligned with ISO/IEC 16262. Changes </tspan> <tspan x="66" y="538" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">between the first and the second </tspan> <tspan x="66" y="557" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle-15" fill="#FFF" d="M58 410h312v111H58z"/><path id="Rectangle-14" fill="#FFF" d="M58 35h312v89H58z"/><path id="Rectangle-1" fill="#DBAF88" d="M395 123v290H21V123h374zm-25 25H46v240h324V148z"/><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M45 22h326v502H45z"/><text id="scrollHeight:723px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" transform="rotate(-90 426 270.5)"><tspan x="350.4" y="275" fill="#C06334">scrollHeight:</tspan> <tspan x="459.6" y="275" fill="#1C85B5">723px</tspan></text><path id="Line-27" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 22h78"/><path id="Line-26" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M371 524h78"/><path id="Line-25" fill="#C06334" fill-rule="nonzero" d="M439 26l7 14h-6v466h6l-7 14-7-14h6V40h-6l7-14z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M335.36 102l14 7-14 7-.001-6H64v6l-14-7 14-7v6h271.359l.001-6z"/><path id="Line-42" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M352 154v-54"/><path id="Line-43" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M47 154v-54"/><text id="scrollWidth-=-324px" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="149.2" y="105" fill="#C06334">scrollWidth = </tspan> <tspan x="266.8" y="105" fill="#1C85B5">324px</tspan></text><g id="Group-2" transform="translate(354 148)"><rect id="Rectangle-19" width="15" height="239" x=".5" y=".5" fill="#D1CFCD" stroke="#D1CFCD" rx="3"/><g id="Rectangle-18-+-Triangle-1"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-2" transform="matrix(1 0 0 -1 0 240)"><rect id="Rectangle-18" width="15" height="19" x=".5" y=".5" fill="url(#linearGradient-1)" stroke="#D1CFCD" rx="3"/><path id="Triangle-1" fill="#7E7C7B" d="M8 7l3.2 6H4.8z"/></g><g id="Rectangle-18-+-Triangle-3-+-Group" transform="translate(0 50)"><g id="Rectangle-18-+-Triangle-3" fill="url(#linearGradient-2)" stroke="#D1CFCD" transform="matrix(1 0 0 -1 0 51)"><rect id="Rectangle-18" width="15" height="50" x=".5" y=".5" rx="3"/></g><g id="Group" fill="#D1CFCD" stroke="#7E7C7B" transform="translate(4 20)"><path id="Rectangle-22" d="M.5.5h7v1h-7z"/><path id="Rectangle-23" d="M.5 3.5h7v1h-7z"/><path id="Rectangle-24" d="M.5 6.5h7v1h-7z"/><path id="Rectangle-25" d="M.5 9.5h7v1h-7z"/></g></g></g></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/10-size-and-scroll-window/article.md b/2-ui/1-document/10-size-and-scroll-window/article.md index a6244b42a..08a2f6576 100644 --- a/2-ui/1-document/10-size-and-scroll-window/article.md +++ b/2-ui/1-document/10-size-and-scroll-window/article.md @@ -1,12 +1,12 @@ # Window sizes and scrolling -How to find out the width and height of the browser window? How to get the full width and height of the document, including the scrolled out part? How to scroll the page using JavaScript? +How do we find the width and height of the browser window? How do we get the full width and height of the document, including the scrolled out part? How do we scroll the page using JavaScript? -From the DOM point of view, the root document element is `document.documentElement`. That element corresponds to `<html>` and has geometry properties described in the [previous chapter](info:size-and-scroll). For some cases we can use it, but there are additional methods and peculiarities important enough to consider. +For this type of information, we can use the root document element `document.documentElement`, that corresponds to the `<html>` tag. But there are additional methods and peculiarities to consider. ## Width/height of the window -Properties `clientWidth/clientHeight` of `document.documentElement` is exactly what we want here: +To get window width and height, we can use the `clientWidth/clientHeight` of `document.documentElement`:  @@ -16,12 +16,12 @@ For instance, this button shows the height of your window: <button onclick="alert(document.documentElement.clientHeight)">alert(document.documentElement.clientHeight)</button> ``` -````warn header="Not `window.innerWidth/Height`" -Browsers also support properties `window.innerWidth/innerHeight`. They look like what we want. So what's the difference? +````warn header="Not `window.innerWidth/innerHeight`" +Browsers also support properties like `window.innerWidth/innerHeight`. They look like what we want, so why not to use them instead? -If there's a scrollbar occupying some space, `clientWidth/clientHeight` provide the width/height inside it. In other words, they return width/height of the visible part of the document, available for the content. +If there exists a scrollbar, and it occupies some space, `clientWidth/clientHeight` provide the width/height without it (subtract it). In other words, they return the width/height of the visible part of the document, available for the content. -And `window.innerWidth/innerHeight` ignore the scrollbar. +`window.innerWidth/innerHeight` includes the scrollbar. If there's a scrollbar, and it occupies some space, then these two lines show different values: ```js run @@ -29,22 +29,22 @@ alert( window.innerWidth ); // full window width alert( document.documentElement.clientWidth ); // window width minus the scrollbar ``` -In most cases we need the *available* window width: to draw or position something. That is: inside scrollbars if there are any. So we should use `documentElement.clientHeight/Width`. +In most cases, we need the *available* window width in order to draw or position something within scrollbars (if there are any), so we should use `documentElement.clientHeight/clientWidth`. ```` ```warn header="`DOCTYPE` is important" Please note: top-level geometry properties may work a little bit differently when there's no `<!DOCTYPE HTML>` in HTML. Odd things are possible. -In modern HTML we should always write `DOCTYPE`. Generally that's not a JavaScript question, but here it affects JavaScript as well. +In modern HTML we should always write `DOCTYPE`. ``` ## Width/height of the document -Theoretically, as the root document element is `documentElement.clientWidth/Height`, and it encloses all the content, we could measure its full size as `documentElement.scrollWidth/scrollHeight`. +Theoretically, as the root document element is `document.documentElement`, and it encloses all the content, we could measure the document's full size as `document.documentElement.scrollWidth/scrollHeight`. -These properties work well for regular elements. But for the whole page these properties do not work as intended. In Chrome/Safari/Opera if there's no scroll, then `documentElement.scrollHeight` may be even less than `documentElement.clientHeight`! For regular elements that's a nonsense. +But on that element, for the whole page, these properties do not work as intended. In Chrome/Safari/Opera, if there's no scroll, then `documentElement.scrollHeight` may be even less than `documentElement.clientHeight`! Weird, right? -To have a reliable result on the full document height, we should take the maximum of these properties: +To reliably obtain the full document height, we should take the maximum of these properties: ```js run let scrollHeight = Math.max( @@ -60,11 +60,11 @@ Why so? Better don't ask. These inconsistencies come from ancient times, not a " ## Get the current scroll [#page-scroll] -Regular elements have their current scroll state in `elem.scrollLeft/scrollTop`. +DOM elements have their current scroll state in their `scrollLeft/scrollTop` properties. -What's with the page? Most browsers provide `documentElement.scrollLeft/Top` for the document scroll, but Chrome/Safari/Opera have bugs (like [157855](https://code.google.com/p/chromium/issues/detail?id=157855), [106133](https://bugs.webkit.org/show_bug.cgi?id=106133)) and we should use `document.body` instead of `document.documentElement` there. +For document scroll, `document.documentElement.scrollLeft/scrollTop` works in most browsers, except older WebKit-based ones, like Safari (bug [5991](https://bugs.webkit.org/show_bug.cgi?id=5991)), where we should use `document.body` instead of `document.documentElement`. -Luckily, we don't have to remember these peculiarities at all, because of the special properties `window.pageXOffset/pageYOffset`: +Luckily, we don't have to remember these peculiarities at all, because the scroll is available in the special properties, `window.pageXOffset/pageYOffset`: ```js run alert('Current scroll from the top: ' + window.pageYOffset); @@ -73,30 +73,34 @@ alert('Current scroll from the left: ' + window.pageXOffset); These properties are read-only. +```smart header="Also available as `window` properties `scrollX` and `scrollY`" +For historical reasons, both properties exist, but they are the same: +- `window.pageXOffset` is an alias of `window.scrollX`. +- `window.pageYOffset` is an alias of `window.scrollY`. +``` + ## Scrolling: scrollTo, scrollBy, scrollIntoView [#window-scroll] ```warn -To scroll the page from JavaScript, its DOM must be fully built. +To scroll the page with JavaScript, its DOM must be fully built. -For instance, if we try to scroll the page from the script in `<head>`, it won't work. +For instance, if we try to scroll the page with a script in `<head>`, it won't work. ``` Regular elements can be scrolled by changing `scrollTop/scrollLeft`. -We can do the same for the page: -- For all browsers except Chrome/Safari/Opera: modify `document.documentElement.scrollTop/Left`. -- In Chrome/Safari/Opera: use `document.body.scrollTop/Left` instead. +We can do the same for the page using `document.documentElement.scrollTop/scrollLeft` (except Safari, where `document.body.scrollTop/Left` should be used instead). -It should work, but smells like cross-browser incompatibilities. Not good. Fortunately, there's a simpler, more universal solution: special methods [window.scrollBy(x,y)](mdn:api/Window/scrollBy) and [window.scrollTo(pageX,pageY)](mdn:api/Window/scrollTo). +Alternatively, there's a simpler, universal solution: special methods [window.scrollBy(x,y)](mdn:api/Window/scrollBy) and [window.scrollTo(pageX,pageY)](mdn:api/Window/scrollTo). -- The method `scrollBy(x,y)` scrolls the page relative to its current position. For instance, `scrollBy(0,10)` scrolls the page `10px` down. +- The method `scrollBy(x,y)` scrolls the page *relative to its current position*. For instance, `scrollBy(0,10)` scrolls the page `10px` down. ```online The button below demonstrates this: <button onclick="window.scrollBy(0,10)">window.scrollBy(0,10)</button> ``` -- The method `scrollTo(pageX,pageY)` scrolls the page relative to the document's top-left corner. It's like setting `scrollLeft/scrollTop`. +- The method `scrollTo(pageX,pageY)` scrolls the page *to absolute coordinates*, so that the top-left corner of the visible part has coordinates `(pageX, pageY)` relative to the document's top-left corner. It's like setting `scrollLeft/scrollTop`. To scroll to the very beginning, we can use `scrollTo(0,0)`. @@ -108,50 +112,50 @@ These methods work for all browsers the same way. ## scrollIntoView -For completeness, let's cover one more method: [elem.scrollIntoView(top)](mdn:api/Element/scrollIntoView). +For completeness, let's cover one more method: [elem.scrollIntoView(top)](mdn:api/Element/scrollIntoView). The call to `elem.scrollIntoView(top)` scrolls the page to make `elem` visible. It has one argument: -- if `top=true` (that's the default), then the page will be scrolled to make `elem` appear on the top of the window. The upper edge of the element is aligned with the window top. -- if `top=false`, then the page scrolls to make `elem` appear at the bottom. The bottom edge of the element is aligned with the window bottom. +- If `top=true` (that's the default), then the page will be scrolled to make `elem` appear on the top of the window. The upper edge of the element will be aligned with the window top. +- If `top=false`, then the page scrolls to make `elem` appear at the bottom. The bottom edge of the element will be aligned with the window bottom. ```online -The button below scrolls the page to make itself show at the window top: +The button below scrolls the page to position itself at the window top: <button onclick="this.scrollIntoView()">this.scrollIntoView()</button> -And this button scrolls the page to show it at the bottom: +And this button scrolls the page to position itself at the bottom: <button onclick="this.scrollIntoView(false)">this.scrollIntoView(false)</button> ``` ## Forbid the scrolling -Sometimes we need to make the document "unscrollable". For instance, when we need to cover it with a large message requiring immediate attention, and we want the visitor to interact with that message, not with the document. +Sometimes we need to make the document "unscrollable". For instance, when we need to cover the page with a large message requiring immediate attention, and we want the visitor to interact with that message, not with the document. -To make the document unscrollable, it's enough to set `document.body.style.overflow = "hidden"`. The page will freeze on its current scroll. +To make the document unscrollable, it's enough to set `document.body.style.overflow = "hidden"`. The page will "freeze" at its current scroll position. ```online Try it: -<button onclick="document.body.style.overflow = 'hidden'">`document.body.style.overflow = 'hidden'`</button> +<button onclick="document.body.style.overflow = 'hidden'">document.body.style.overflow = 'hidden'</button> -<button onclick="document.body.style.overflow = ''">`document.body.style.overflow = ''`</button> +<button onclick="document.body.style.overflow = ''">document.body.style.overflow = ''</button> -The first button freezes the scroll, the second one resumes it. +The first button freezes the scroll, while the second one releases it. ``` -We can use the same technique to "freeze" the scroll for other elements, not just for `document.body`. +We can use the same technique to freeze the scroll for other elements, not just for `document.body`. -The drawback of the method is that the scrollbar disappears. If it occupied some space, then that space is now free, and the content "jumps" to fill it. +The drawback of the method is that the scrollbar disappears. If it occupied some space, then that space is now free and the content "jumps" to fill it. -That looks a bit odd, but can be worked around if we compare `clientWidth` before and after the freeze, and if it increased (the scrollbar disappeared) then add `padding` to `document.body` in place of the scrollbar, to keep the content width the same. +That looks a bit odd, but can be worked around if we compare `clientWidth` before and after the freeze. If it increased (the scrollbar disappeared), then add `padding` to `document.body` in place of the scrollbar to keep the content width the same. ## Summary Geometry: -- Width/height of the visible part of the document (content area width/height): `document.documentElement.clientWidth/Height` +- Width/height of the visible part of the document (content area width/height): `document.documentElement.clientWidth/clientHeight` - Width/height of the whole document, with the scrolled out part: ```js diff --git a/2-ui/1-document/10-size-and-scroll-window/document-client-width-height.svg b/2-ui/1-document/10-size-and-scroll-window/document-client-width-height.svg index 65e77ae80..0add5cbdd 100644 --- a/2-ui/1-document/10-size-and-scroll-window/document-client-width-height.svg +++ b/2-ui/1-document/10-size-and-scroll-window/document-client-width-height.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="508" height="203" viewBox="0 0 508 203"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="document-client-width-height.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(275 8)"><path id="Shape" d="M179.188 145H3.813C1.708 145 0 143.369 0 141.375V3.625C0 1.631 1.708 0 3.813 0h175.375C181.284 0 183 1.631 183 3.625v137.75c0 1.994-1.716 3.625-3.813 3.625zM7.625 137.75h167.75V7.25H7.625v130.5z"/><path id="Shape" d="M175.375 36.25H7.625c-2.104 0-3.813-1.631-3.813-3.625S5.521 29 7.626 29h167.75c2.097 0 3.813 1.631 3.813 3.625s-1.716 3.625-3.813 3.625zM61.32 126.578c-1.266 0-2.508-.595-3.233-1.682L34.892 89.849a3.486 3.486 0 01.16-4.082l23.195-29.906c1.25-1.631 3.63-1.964 5.33-.746 1.693 1.188 2.059 3.458.8 5.067l-21.646 27.92 21.807 32.951c1.12 1.69.587 3.937-1.197 4.988a3.936 3.936 0 01-2.02.537zM122 126.578a3.802 3.802 0 01-2.028-.559c-1.784-1.058-2.326-3.298-1.197-4.988l21.807-32.95-21.655-27.928c-1.25-1.617-.892-3.878.8-5.067 1.694-1.197 4.08-.849 5.33.76l23.188 29.907a3.462 3.462 0 01.16 4.082L125.21 124.88c-.709 1.102-1.944 1.697-3.21 1.697zM76.25 119.937a4.07 4.07 0 01-1.86-.457c-1.838-.979-2.494-3.183-1.465-4.959l30.18-51.359c1.03-1.755 3.34-2.385 5.2-1.37 1.83.978 2.486 3.175 1.457 4.937l-30.18 51.359c-.701 1.182-1.998 1.849-3.332 1.849zM22.875 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM38.125 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM53.375 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625z"/></g><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="101">documentElement.clientHeight</tspan></text><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="257" y="195">documentElement.clientWidth</tspan></text><path id="Line-3-Copy" fill="#C06334" fill-rule="nonzero" d="M431.025 158.653l19 9.5-19 9.5-.001-8h-132v8l-19-9.5 19-9.5v8h132v-8z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M256.5 45l9.5 19-8-.001v64l8 .001-9.5 19-9.5-19 8-.001v-64L247 64l9.5-19z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="508" height="203" viewBox="0 0 508 203"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="document-client-width-height.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(275 8)"><path id="Shape" d="M179.188 145H3.813C1.708 145 0 143.369 0 141.375V3.625C0 1.631 1.708 0 3.813 0h175.375C181.284 0 183 1.631 183 3.625v137.75c0 1.994-1.716 3.625-3.813 3.625zM7.625 137.75h167.75V7.25H7.625v130.5z"/><path id="Shape" d="M175.375 36.25H7.625c-2.104 0-3.813-1.631-3.813-3.625S5.521 29 7.626 29h167.75c2.097 0 3.813 1.631 3.813 3.625s-1.716 3.625-3.813 3.625zM61.32 126.578c-1.266 0-2.508-.595-3.233-1.682L34.892 89.849a3.486 3.486 0 01.16-4.082l23.195-29.906c1.25-1.631 3.63-1.964 5.33-.746 1.693 1.188 2.059 3.458.8 5.067l-21.646 27.92 21.807 32.951c1.12 1.69.587 3.937-1.197 4.988a3.936 3.936 0 01-2.02.537zM122 126.578a3.802 3.802 0 01-2.028-.559c-1.784-1.058-2.326-3.298-1.197-4.988l21.807-32.95-21.655-27.928c-1.25-1.617-.892-3.878.8-5.067 1.694-1.197 4.08-.849 5.33.76l23.188 29.907a3.462 3.462 0 01.16 4.082L125.21 124.88c-.709 1.102-1.944 1.697-3.21 1.697zM76.25 119.937a4.07 4.07 0 01-1.86-.457c-1.838-.979-2.494-3.183-1.465-4.959l30.18-51.359c1.03-1.755 3.34-2.385 5.2-1.37 1.83.978 2.486 3.175 1.457 4.937l-30.18 51.359c-.701 1.182-1.998 1.849-3.332 1.849zM22.875 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM38.125 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM53.375 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625z"/></g><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="101">documentElement.clientHeight</tspan></text><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="257" y="195">documentElement.clientWidth</tspan></text><path id="Line-3-Copy" fill="#C06334" fill-rule="nonzero" d="M431.025 158.653l19 9.5-19 9.5-.001-8h-132v8l-19-9.5 19-9.5v8h132v-8z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M256.5 45l9.5 19-8-.001v64l8 .001-9.5 19-9.5-19 8-.001v-64L247 64l9.5-19z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="508" height="203" viewBox="0 0 508 203"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="document-client-width-height.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(275 8)"><path id="Shape" d="M179.188 145H3.813C1.708 145 0 143.369 0 141.375V3.625C0 1.631 1.708 0 3.813 0h175.375C181.284 0 183 1.631 183 3.625v137.75c0 1.994-1.716 3.625-3.813 3.625zM7.625 137.75h167.75V7.25H7.625v130.5z"/><path id="Shape" d="M175.375 36.25H7.625c-2.104 0-3.813-1.631-3.813-3.625S5.521 29 7.626 29h167.75c2.097 0 3.813 1.631 3.813 3.625s-1.716 3.625-3.813 3.625zM61.32 126.578c-1.266 0-2.508-.595-3.233-1.682L34.892 89.849a3.486 3.486 0 01.16-4.082l23.195-29.906c1.25-1.631 3.63-1.964 5.33-.746 1.693 1.188 2.059 3.458.8 5.067l-21.646 27.92 21.807 32.951c1.12 1.69.587 3.937-1.197 4.988a3.936 3.936 0 01-2.02.537zM122 126.578a3.802 3.802 0 01-2.028-.559c-1.784-1.058-2.326-3.298-1.197-4.988l21.807-32.95-21.655-27.928c-1.25-1.617-.892-3.878.8-5.067 1.694-1.197 4.08-.849 5.33.76l23.188 29.907a3.462 3.462 0 01.16 4.082L125.21 124.88c-.709 1.102-1.944 1.697-3.21 1.697zM76.25 119.937a4.07 4.07 0 01-1.86-.457c-1.838-.979-2.494-3.183-1.465-4.959l30.18-51.359c1.03-1.755 3.34-2.385 5.2-1.37 1.83.978 2.486 3.175 1.457 4.937l-30.18 51.359c-.701 1.182-1.998 1.849-3.332 1.849zM22.875 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM38.125 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625zM53.375 18.125c0 2.003-1.706 3.625-3.813 3.625-2.106 0-3.812-1.622-3.812-3.625s1.706-3.625 3.813-3.625c2.106 0 3.812 1.622 3.812 3.625z"/></g><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="12" y="101">documentElement.clientHeight</tspan></text><text id="documentElement.clie" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="257" y="195">documentElement.clientWidth</tspan></text><path id="Line-3-Copy" fill="#C06334" fill-rule="nonzero" d="M299.025 177.653l-19-9.5 19-9.5-.001 8h132v-8l19 9.5-19 9.5v-8h-132v8z" transform="matrix(-1 0 0 1 730.025 0)"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M256.5 45l9.5 19-8-.001v64l8 .001-9.5 19-9.5-19 8-.001v-64L247 64l9.5-19z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/1-document/11-coordinates/1-find-point-coordinates/solution.md b/2-ui/1-document/11-coordinates/1-find-point-coordinates/solution.md index f3547096f..4101d4915 100644 --- a/2-ui/1-document/11-coordinates/1-find-point-coordinates/solution.md +++ b/2-ui/1-document/11-coordinates/1-find-point-coordinates/solution.md @@ -1,6 +1,6 @@ # Outer corners -Outer corners are basically what we get from [elem.getBoundingClientRect()](https://developer.mozilla.org/en-US/docs/DOM/element.getBoundingClientRect). +Outer corners are basically what we get from [elem.getBoundingClientRect()](https://developer.mozilla.org/en-US/docs/DOM/element.getBoundingClientRect). Coordinates of the upper-left corner `answer1` and the bottom-right corner `answer2`: diff --git a/2-ui/1-document/11-coordinates/2-position-at/source.view/index.html b/2-ui/1-document/11-coordinates/2-position-at/source.view/index.html index 730f815d9..675573450 100755 --- a/2-ui/1-document/11-coordinates/2-position-at/source.view/index.html +++ b/2-ui/1-document/11-coordinates/2-position-at/source.view/index.html @@ -41,7 +41,13 @@ * relative to the anchor element. */ function showNote(anchor, position, html) { - // ... your code ... + + let note = document.createElement('div'); + note.className = "note"; + note.innerHTML = html; + document.body.append(note); + + positionAt(anchor, position, note); } // test it diff --git a/2-ui/1-document/11-coordinates/2-position-at/task.md b/2-ui/1-document/11-coordinates/2-position-at/task.md index 1d4f06771..3aaa47f03 100644 --- a/2-ui/1-document/11-coordinates/2-position-at/task.md +++ b/2-ui/1-document/11-coordinates/2-position-at/task.md @@ -4,12 +4,15 @@ importance: 5 # Show a note near the element -Create a function `positionAt(anchor, position, elem)` that positions `elem`, depending on `position` either at the top (`"top"`), right (`"right"`) or bottom (`"bottom"`) of the element `anchor`. +Create a function `positionAt(anchor, position, elem)` that positions `elem`, depending on `position` near `anchor` element. -Call it inside the function `showNote(anchor, position, html)` that shows an element with the class `"note"` and the text `html` at the given position near the anchor. +The `position` must be a string with any one of 3 values: +- `"top"` - position `elem` right above `anchor` +- `"right"` - position `elem` immediately at the right of `anchor` +- `"bottom"` - position `elem` right below `anchor` -Show the notes like here: +It's used inside function `showNote(anchor, position, html)`, provided in the task source code, that creates a "note" element with given `html` and shows it at the given `position` near the `anchor`. -[iframe src="solution" height="350" border="1" link] +Here's the demo of notes: -P.S. The note should have `position:fixed` for this task. +[iframe src="solution" height="350" border="1" link] diff --git a/2-ui/1-document/11-coordinates/3-position-at-absolute/solution.view/index.html b/2-ui/1-document/11-coordinates/3-position-at-absolute/solution.view/index.html index d627bee84..56c95d5ec 100644 --- a/2-ui/1-document/11-coordinates/3-position-at-absolute/solution.view/index.html +++ b/2-ui/1-document/11-coordinates/3-position-at-absolute/solution.view/index.html @@ -28,8 +28,8 @@ let box = elem.getBoundingClientRect(); return { - top: box.top + pageYOffset, - left: box.left + pageXOffset + top: box.top + window.pageYOffset, + left: box.left + window.pageXOffset }; } diff --git a/2-ui/1-document/11-coordinates/4-position-inside-absolute/solution.view/index.html b/2-ui/1-document/11-coordinates/4-position-inside-absolute/solution.view/index.html index 7e841397b..b89db3790 100644 --- a/2-ui/1-document/11-coordinates/4-position-inside-absolute/solution.view/index.html +++ b/2-ui/1-document/11-coordinates/4-position-inside-absolute/solution.view/index.html @@ -26,8 +26,8 @@ let box = elem.getBoundingClientRect(); return { - top: box.top + pageYOffset, - left: box.left + pageXOffset + top: box.top + window.pageYOffset, + left: box.left + window.pageXOffset }; } diff --git a/2-ui/1-document/11-coordinates/article.md b/2-ui/1-document/11-coordinates/article.md index 61b5161eb..fc605c414 100644 --- a/2-ui/1-document/11-coordinates/article.md +++ b/2-ui/1-document/11-coordinates/article.md @@ -4,58 +4,102 @@ To move elements around we should be familiar with coordinates. Most JavaScript methods deal with one of two coordinate systems: -1. Relative to the window(or another viewport) top/left. -2. Relative to the document top/left. +1. **Relative to the window** - similar to `position:fixed`, calculated from the window top/left edge. + - we'll denote these coordinates as `clientX/clientY`, the reasoning for such name will become clear later, when we study event properties. +2. **Relative to the document** - similar to `position:absolute` in the document root, calculated from the document top/left edge. + - we'll denote them `pageX/pageY`. -It's important to understand the difference and which type is where. +When the page is scrolled to the very beginning, so that the top/left corner of the window is exactly the document top/left corner, these coordinates equal each other. But after the document shifts, window-relative coordinates of elements change, as elements move across the window, while document-relative coordinates remain the same. -## Window coordinates: getBoundingClientRect +On this picture we take a point in the document and demonstrate its coordinates before the scroll (left) and after it (right): -Window coordinates start at the upper-left corner of the window. + -The method `elem.getBoundingClientRect()` returns window coordinates for `elem` as an object with properties: +When the document scrolled: +- `pageY` - document-relative coordinate stayed the same, it's counted from the document top (now scrolled out). +- `clientY` - window-relative coordinate did change (the arrow became shorter), as the same point became closer to window top. -- `top` -- Y-coordinate for the top element edge, -- `left` -- X-coordinate for the left element edge, -- `right` -- X-coordinate for the right element edge, -- `bottom` -- Y-coordinate for the bottom element edge. +## Element coordinates: getBoundingClientRect -Like this: +The method `elem.getBoundingClientRect()` returns window coordinates for a minimal rectangle that encloses `elem` as an object of built-in [DOMRect](https://www.w3.org/TR/geometry-1/#domrect) class. - +Main `DOMRect` properties: +- `x/y` -- X/Y-coordinates of the rectangle origin relative to window, +- `width/height` -- width/height of the rectangle (can be negative). -Window coordinates do not take the scrolled out part of the document into account, they are calculated from the window's upper-left corner. +Additionally, there are derived properties: -In other words, when we scroll the page, the element goes up or down, *its window coordinates change*. That's very important. +- `top/bottom` -- Y-coordinate for the top/bottom rectangle edge, +- `left/right` -- X-coordinate for the left/right rectangle edge. ```online -Click the button to see its window coordinates: +For instance click this button to see its window coordinates: -<input id="brTest" type="button" value="Show button.getBoundingClientRect() for this button" onclick='showRect(this)'/> +<p><input id="brTest" type="button" style="max-width: 90vw;" value="Get coordinates using button.getBoundingClientRect() for this button" onclick='showRect(this)'/></p> <script> function showRect(elem) { let r = elem.getBoundingClientRect(); - alert("{top:"+r.top+", left:"+r.left+", right:"+r.right+", bottom:"+ r.bottom + "}"); + alert(`x:${r.x} +y:${r.y} +width:${r.width} +height:${r.height} +top:${r.top} +bottom:${r.bottom} +left:${r.left} +right:${r.right} +`); } </script> -If you scroll the page, the button position changes, and window coordinates as well. +If you scroll the page and repeat, you'll notice that as window-relative button position changes, its window coordinates (`y/top/bottom` if you scroll vertically) change as well. ``` -Also: +Here's the picture of `elem.getBoundingClientRect()` output: -- Coordinates may be decimal fractions. That's normal, internally browser uses them for calculations. We don't have to round them when setting to `style.position.left/top`, the browser is fine with fractions. -- Coordinates may be negative. For instance, if the page is scrolled down and the top `elem` is now above the window. Then, `elem.getBoundingClientRect().top` is negative. -- Some browsers (like Chrome) provide additional properties, `width` and `height` of the element that invoked the method to `getBoundingClientRect` as the result. We can also get them by subtraction: `height=bottom-top`, `width=right-left`. + -```warn header="Coordinates right/bottom are different from CSS properties" -If we compare window coordinates versus CSS positioning, then there are obvious similarities to `position:fixed`. The positioning of an element is also relative to the viewport. +As you can see, `x/y` and `width/height` fully describe the rectangle. Derived properties can be easily calculated from them: -But in CSS, the `right` property means the distance from the right edge, and the `bottom` property means the distance from the bottom edge. +- `left = x` +- `top = y` +- `right = x + width` +- `bottom = y + height` -If we just look at the picture above, we can see that in JavaScript it is not so. All window coordinates are counted from the upper-left corner, including these ones. +Please note: + +- Coordinates may be decimal fractions, such as `10.5`. That's normal, internally browser uses fractions in calculations. We don't have to round them when setting to `style.left/top`. +- Coordinates may be negative. For instance, if the page is scrolled so that `elem` is now above the window, then `elem.getBoundingClientRect().top` is negative. + +```smart header="Why derived properties are needed? Why does `top/left` exist if there's `x/y`?" +Mathematically, a rectangle is uniquely defined with its starting point `(x,y)` and the direction vector `(width,height)`. So the additional derived properties are for convenience. + +Technically it's possible for `width/height` to be negative, that allows for "directed" rectangle, e.g. to represent mouse selection with properly marked start and end. + +Negative `width/height` values mean that the rectangle starts at its bottom-right corner and then "grows" left-upwards. + +Here's a rectangle with negative `width` and `height` (e.g. `width=-200`, `height=-100`): + + + +As you can see, `left/top` do not equal `x/y` in such case. + +In practice though, `elem.getBoundingClientRect()` always returns positive width/height, here we mention negative `width/height` only for you to understand why these seemingly duplicate properties are not actually duplicates. +``` + +```warn header="Internet Explorer: no support for `x/y`" +Internet Explorer doesn't support `x/y` properties for historical reasons. + +So we can either make a polyfill (add getters in `DomRect.prototype`) or just use `top/left`, as they are always the same as `x/y` for positive `width/height`, in particular in the result of `elem.getBoundingClientRect()`. +``` + +```warn header="Coordinates right/bottom are different from CSS position properties" +There are obvious similarities between window-relative coordinates and CSS `position:fixed`. + +But in CSS positioning, `right` property means the distance from the right edge, and `bottom` property means the distance from the bottom edge. + +If we just look at the picture above, we can see that in JavaScript it is not so. All window coordinates are counted from the top-left corner, including these ones. ``` ## elementFromPoint(x, y) [#elementFromPoint] @@ -87,8 +131,6 @@ The method `document.elementFromPoint(x,y)` only works if `(x,y)` are inside the If any of the coordinates is negative or exceeds the window width/height, then it returns `null`. -In most cases such behavior is not a problem, but we should keep that in mind. - Here's a typical error that may occur if we don't check for it: ```js @@ -100,11 +142,11 @@ elem.style.background = ''; // Error! ``` ```` -## Using for position:fixed +## Using for "fixed" positioning -Most of time we need coordinates to position something. In CSS, to position an element relative to the viewport we use `position:fixed` together with `left/top` (or `right/bottom`). +Most of time we need coordinates in order to position something. -We can use `getBoundingClientRect` to get coordinates of an element, and then to show something near it. +To show something near an element, we can use `getBoundingClientRect` to get its coordinates, and then CSS `position` together with `left/top` (or `right/bottom`). For instance, the function `createMessageUnder(elem, html)` below shows the message under `elem`: @@ -151,7 +193,7 @@ The reason is obvious: the message element relies on `position:fixed`, so it rem To change that, we need to use document-based coordinates and `position:absolute`. -## Document coordinates +## Document coordinates [#getCoords] Document-relative coordinates start from the upper-left corner of the document, not the window. @@ -159,24 +201,6 @@ In CSS, window coordinates correspond to `position:fixed`, while document coordi We can use `position:absolute` and `top/left` to put something at a certain place of the document, so that it remains there during a page scroll. But we need the right coordinates first. -For clarity we'll call window coordinates `(clientX,clientY)` and document coordinates `(pageX,pageY)`. - -When the page is not scrolled, then window coordinate and document coordinates are actually the same. Their zero points match too: - - - -And if we scroll it, then `(clientX,clientY)` change, because they are relative to the window, but `(pageX,pageY)` remain the same. - -Here's the same page after the vertical scroll: - - - -- `clientY` of the header `"From today's featured article"` became `0`, because the element is now on window top. -- `clientX` didn't change, as we didn't scroll horizontally. -- `pageX` and `pageY` coordinates of the element are still the same, because they are relative to the document. - -## Getting document coordinates [#getCoords] - There's no standard method to get the document coordinates of an element. But it's easy to write it. The two coordinate systems are connected by the formula: @@ -191,12 +215,34 @@ function getCoords(elem) { let box = elem.getBoundingClientRect(); return { - top: box.top + pageYOffset, - left: box.left + pageXOffset + top: box.top + window.pageYOffset, + right: box.right + window.pageXOffset, + bottom: box.bottom + window.pageYOffset, + left: box.left + window.pageXOffset }; } ``` +If in the example above we used it with `position:absolute`, then the message would stay near the element on scroll. + +The modified `createMessageUnder` function: + +```js +function createMessageUnder(elem, html) { + let message = document.createElement('div'); + message.style.cssText = "*!*position:absolute*/!*; color: red"; + + let coords = *!*getCoords(elem);*/!* + + message.style.left = coords.left + "px"; + message.style.top = coords.bottom + "px"; + + message.innerHTML = html; + + return message; +} +``` + ## Summary Any point on the page has coordinates: @@ -206,4 +252,4 @@ Any point on the page has coordinates: Window coordinates are great to use with `position:fixed`, and document coordinates do well with `position:absolute`. -Both coordinate systems have their "pro" and "contra", there are times we need one or the other one, just like CSS `position` `absolute` and `fixed`. +Both coordinate systems have their pros and cons; there are times we need one or the other one, just like CSS `position` `absolute` and `fixed`. diff --git a/2-ui/1-document/11-coordinates/coordinates-negative.svg b/2-ui/1-document/11-coordinates/coordinates-negative.svg new file mode 100644 index 000000000..4f2e78687 --- /dev/null +++ b/2-ui/1-document/11-coordinates/coordinates-negative.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="521" height="355" viewBox="0 0 521 355"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-1" width="118%" height="209.1%" x="-9%" y="-54.5%" filterUnits="objectBoundingBox"><feGaussianBlur in="SourceGraphic" stdDeviation="4"/></filter></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="coordinates-negative.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(13 11)"><path id="Shape" d="M455.313 330H9.688C4.34 330 0 326.288 0 321.75V8.25C0 3.713 4.34 0 9.688 0h445.625C460.64 0 465 3.712 465 8.25v313.5c0 4.538-4.36 8.25-9.688 8.25zM9.432 318.939h444.25V11.06H9.432V318.94z"/><path id="Shape" d="M19.674 46C14.334 46 10 44.875 10 43.5s4.334-2.5 9.674-2.5h425.652c5.32 0 9.674 1.125 9.674 2.5s-4.353 2.5-9.674 2.5H19.674zM31 27.5c0 3.039-2.461 5.5-5.5 5.5a5.498 5.498 0 01-5.5-5.5c0-3.039 2.461-5.5 5.5-5.5s5.5 2.461 5.5 5.5zM59 27.5c0 3.039-2.461 5.5-5.5 5.5a5.498 5.498 0 01-5.5-5.5c0-3.039 2.461-5.5 5.5-5.5s5.5 2.461 5.5 5.5zM87 27.5c0 3.039-2.461 5.5-5.5 5.5a5.498 5.498 0 01-5.5-5.5c0-3.039 2.461-5.5 5.5-5.5s5.5 2.461 5.5 5.5z"/></g><text id="bottom" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="414" y="196">bottom</tspan></text><path id="Line-30-Copy-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M392.5 290v37"/><path id="Line-29-Copy" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M394 288h69"/><text id="(x,y)" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="395" y="305">(x,y)</tspan></text><text id="(x,y)-copy" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="395" y="305">(x,y)</tspan></text><text id="left" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="50" y="129">left</tspan></text><path id="Line-Copy" fill="#7E7C7B" fill-rule="nonzero" d="M410 56.5v213h8l-9.5 19-9.5-19h8v-213h3z"/><text id="right" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="188" y="323">right</tspan></text><path id="Line-3-Copy-2" fill="#7E7C7B" fill-rule="nonzero" d="M371.498 295.5l19 9.5-19 9.5-.001-8H21.502v-3h349.995v-8z"/><path id="Rectangle-1" fill="#DBAF88" d="M393 113v175H118V113h275zm-10.643 10H129.015v155h253.342V123z"/><text id="Introduction-This-Ec" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold" opacity=".7"><tspan x="139" y="148">Introduction</tspan> <tspan x="139" y="176" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on </tspan> <tspan x="139" y="195" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">several originating technologies, </tspan> <tspan x="139" y="214" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">the most well known being </tspan> <tspan x="139" y="233" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JavaScript (Netscape) and JScript </tspan> <tspan x="139" y="252" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">(Microsoft). The language was </tspan> <tspan x="139" y="271" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at </tspan></text><text id="top" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="125" y="90">top</tspan></text><path id="Line-3" fill="#7E7C7B" fill-rule="nonzero" d="M100.548 104.403l18.95 9.6-19.05 9.4.041-8.001-76.5-.402-1.5-.008.017-3 1.5.008 76.499.402.043-8z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M118.076 113.232l21.132 2.162-4.288 6.754 255.882 162.436 1.266.804-1.608 2.532-1.266-.804-255.882-162.435-4.287 6.754-10.95-18.203z"/><path id="Line-3-Copy" fill="#7E7C7B" fill-rule="nonzero" d="M118.998 56.5V58L119 94.499h8l-9.498 19.001-9.501-19H116l-.002-36.5v-1.5h3z"/><g id="Group" transform="rotate(32.5 -148.357 454.177)"><path id="Rectangle" fill="#FFF" d="M0 0h133v22H0z" filter="url(#filter-1)"/><text id="(width,height)" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="8.682" y="15.378">(width,height)</tspan></text></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/11-coordinates/coordinates.svg b/2-ui/1-document/11-coordinates/coordinates.svg new file mode 100644 index 000000000..261ff6696 --- /dev/null +++ b/2-ui/1-document/11-coordinates/coordinates.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="521" height="411" viewBox="0 0 521 411"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="coordinates.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(13 11)"><path id="Shape" d="M490.563 386H10.438C4.676 386 0 381.658 0 376.35V9.65C0 4.342 4.676 0 10.438 0h480.125C496.303 0 501 4.343 501 9.65v366.7c0 5.308-4.697 9.65-10.438 9.65zm-480.4-12.939h478.642V12.94H10.162V373.06z"/><path id="Shape" d="M20.859 54.1c-5.753 0-10.422-1.147-10.422-2.55 0-1.402 4.67-2.55 10.422-2.55H479.4c5.732 0 10.422 1.148 10.422 2.55 0 1.403-4.69 2.55-10.422 2.55H20.86zM33.8 31.627a6.024 6.024 0 11-12.05 0 6.024 6.024 0 1112.05 0zM63.988 31.627a6.024 6.024 0 11-12.05 0 6.024 6.024 0 1112.05 0zM94.177 31.627a6.024 6.024 0 11-12.05 0 6.024 6.024 0 1112.05 0z"/></g><text id="height" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="417" y="208">height</tspan></text><text id="bottom" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="451" y="311">bottom</tspan></text><path id="Line-28" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M395 293h103"/><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M392 296v74.5"/><path id="Line-30-Copy" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M119 296v46"/><path id="Line-29" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M396 120h30"/><text id="x" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="62" y="113">x</tspan></text><text id="left" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="50" y="135">left</tspan></text><text id="y" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="125" y="79">y</tspan></text><text id="width" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="234" y="339">width</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M414 118.5V273h8l-9.5 19-9.5-19h8V118.5h3z"/><path id="Line-Copy" fill="#7E7C7B" fill-rule="nonzero" d="M476 65v208h8l-9.5 19-9.5-19h8V65h3z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M371 309l19 9.5-19 9.5-.001-8H116v-3h254.999l.001-8z"/><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M100.05 109.903l18.95 9.6-19.05 9.4.042-8.001-76.5-.402-1.5-.008.016-3 1.5.008 76.5.402.042-8z"/><text id="right" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="180" y="375">right</tspan></text><path id="Line-3-Copy-2" fill="#7E7C7B" fill-rule="nonzero" d="M371 350l19 9.5-19 9.5v-8H21.005v-3H371v-8z"/><path id="Line-3-Copy" fill="#C06334" fill-rule="nonzero" d="M119 65v1.5l.003 34.249h8l-9.498 19.001-9.502-19h8L116 66.5V65h3z" transform="matrix(-1 0 0 1 235.005 0)"/><path id="Rectangle-1" fill="#DBAF88" d="M392.629 119v175H118V119h274.629zM382 129H129v155h253V129z"/><text id="Introduction-This-Ec" fill="#643B0C" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold" opacity=".8"><tspan x="138.946" y="153.8">Introduction</tspan> <tspan x="138.946" y="181.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on </tspan> <tspan x="138.946" y="200.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">several originating technologies, </tspan> <tspan x="138.946" y="219.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">the most well known being </tspan> <tspan x="138.946" y="238.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JavaScript (Netscape) and JScript </tspan> <tspan x="138.946" y="257.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">(Microsoft). The language was </tspan> <tspan x="138.946" y="276.8" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at </tspan></text><text id="top" fill="#7E7C7B" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="125" y="96">top</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/11-coordinates/coords.png b/2-ui/1-document/11-coordinates/coords.png deleted file mode 100644 index 79d1fdbb6..000000000 Binary files a/2-ui/1-document/11-coordinates/coords.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/coords@2x.png b/2-ui/1-document/11-coordinates/coords@2x.png deleted file mode 100644 index d3e340f50..000000000 Binary files a/2-ui/1-document/11-coordinates/coords@2x.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/document-and-window-coordinates-scrolled.svg b/2-ui/1-document/11-coordinates/document-and-window-coordinates-scrolled.svg new file mode 100644 index 000000000..f03317f0c --- /dev/null +++ b/2-ui/1-document/11-coordinates/document-and-window-coordinates-scrolled.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="728" height="358" viewBox="0 0 728 358"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="document-and-window-coordinates-scrolled.svg"><path fill="#FFF" d="M0 0h728v358H0z"/><text id="Introduction-This-Ec" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold" opacity=".5"><tspan x="34" y="176">Introduction</tspan> <tspan x="34" y="204" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="34" y="223" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="34" y="242" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="34" y="261" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="34" y="280" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="34" y="299" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="34" y="318" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. </tspan> <tspan x="34" y="337" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> </tspan> <tspan x="34" y="356" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">It has appeared in all subsequent browsers </tspan> <tspan x="34" y="375" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">from Netscape and in all browsers from </tspan> <tspan x="34" y="394" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft starting with Internet Explorer </tspan> <tspan x="34" y="413" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">3.0. The development of this Standard </tspan> <tspan x="34" y="432" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">started in November 1996. </tspan> <tspan x="34" y="451" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The first edition of this Ecma Standard was </tspan> <tspan x="34" y="470" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">adopted by the Ecma General Assembly of </tspan> <tspan x="34" y="489" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">June 1997.</tspan> <tspan x="34" y="508" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="34" y="527" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="34" y="546" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="34" y="565" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="34" y="584" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="34" y="603" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="34" y="622" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="34" y="641" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="34" y="660" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><text id="😍" fill="#643B0C" font-family="AppleColorEmoji, Apple Color Emoji" font-size="18" font-weight="normal"><tspan x="139" y="330">😍</tspan></text><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M150.5 131v167h6l-7 14-7-14h6V131h2z"/><text id="pageY" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="157" y="155">pageY</tspan></text><text id="clientY" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="156.4" y="177">clientY</tspan></text><path id="Line-39-Copy" fill="#C06334" fill-rule="nonzero" d="M126 315.5l14 7-14 7-.001-6.001L13 323.5h-1v-2h1l112.999-.001.001-5.999z"/><text id="pageX" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="23" y="316">pageX</tspan></text><text id="clientX" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="22.4" y="337">clientX</tspan></text><g id="np_browser_551045_E8C48F" fill="#DBAF88" fill-rule="nonzero" transform="translate(-1 68)"><path id="Shape" d="M22.8 0C7.76 0 0 8.157 0 24v242c0 15.843 7.759 24 22.8 24h309.346c15.042 0 21.854-7.157 21.854-23V24c0-15.843-6.812-24-21.854-24H22.8zm-5.425 15.286h318.177c2.535 0 4.187 2.012 4.187 5.1L338 55.328H13.188V20.386c0-3.088 1.652-5.1 4.187-5.1zm21.159 13.857c-3.902 0-7.067 3.327-7.067 7.428S34.632 44 38.534 44s7.067-3.328 7.067-7.429c0-4.1-3.165-7.428-7.067-7.428zm31.467 0c-3.902 0-7.068 3.327-7.068 7.428S66.1 44 70.001 44c3.901 0 7.067-3.328 7.067-7.429 0-4.1-3.166-7.428-7.067-7.428zm31.466 0c-3.901 0-7.067 3.327-7.067 7.428S97.566 44 101.467 44c3.902 0 7.068-3.328 7.068-7.429 0-4.1-3.166-7.428-7.068-7.428zM14 71.172h324l1.865 198.376c0 2.782-1.651 4.595-4.186 4.595H17.5c-2.535 0-4.187-1.813-4.187-4.595L14 71.172z"/></g><text id="Introduction-This-Ec" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold" opacity=".5"><tspan x="409" y="34">Introduction</tspan> <tspan x="409" y="62" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">This Ecma Standard is based on several </tspan> <tspan x="409" y="81" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">originating technologies, the most well </tspan> <tspan x="409" y="100" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">known being JavaScript (Netscape) and </tspan> <tspan x="409" y="119" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">JScript (Microsoft). The language was </tspan> <tspan x="409" y="138" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">invented by Brendan Eich at Netscape and </tspan> <tspan x="409" y="157" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">first appeared in that company’s Navigator </tspan> <tspan x="409" y="176" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">2.0 browser. </tspan> <tspan x="409" y="195" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> </tspan> <tspan x="409" y="214" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">It has appeared in all subsequent browsers </tspan> <tspan x="409" y="233" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">from Netscape and in all browsers from </tspan> <tspan x="409" y="252" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Microsoft starting with Internet Explorer </tspan> <tspan x="409" y="271" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">3.0. The development of this Standard </tspan> <tspan x="409" y="290" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">started in November 1996. </tspan> <tspan x="409" y="309" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">The first edition of this Ecma Standard was </tspan> <tspan x="409" y="328" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">adopted by the Ecma General Assembly of </tspan> <tspan x="409" y="347" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">June 1997.</tspan> <tspan x="409" y="366" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">That Ecma Standard was submitted to ISO/</tspan> <tspan x="409" y="385" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">IEC JTC 1 for adoption under the fast-track </tspan> <tspan x="409" y="404" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">procedure, and approved as international </tspan> <tspan x="409" y="423" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">standard ISO/IEC 16262, in April 1998. The </tspan> <tspan x="409" y="442" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Ecma General Assembly of June 1998 </tspan> <tspan x="409" y="461" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">approved the second edition of ECMA-262 </tspan> <tspan x="409" y="480" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">to keep it fully aligned with ISO/IEC 16262. </tspan> <tspan x="409" y="499" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">Changes between the first and the second </tspan> <tspan x="409" y="518" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal">edition are editorial in nature.</tspan></text><path id="Rectangle" fill="#FFF" d="M392 87h307v42H392z" opacity=".6"/><g id="np_browser_551045_E8C48F" fill="#DBAF88" fill-rule="nonzero" transform="translate(374 71)"><path id="Shape" d="M22.8 0C7.76 0 0 8.157 0 24v242c0 15.843 7.759 24 22.8 24h309.346c15.042 0 21.854-7.157 21.854-23V24c0-15.843-6.812-24-21.854-24H22.8zm-5.425 15.286h318.177c2.535 0 4.187 2.012 4.187 5.1L338 55.328H13.188V20.386c0-3.088 1.652-5.1 4.187-5.1zm21.159 13.857c-3.902 0-7.067 3.327-7.067 7.428S34.632 44 38.534 44s7.067-3.328 7.067-7.429c0-4.1-3.165-7.428-7.067-7.428zm31.467 0c-3.902 0-7.068 3.327-7.068 7.428S66.1 44 70.001 44c3.901 0 7.067-3.328 7.067-7.429 0-4.1-3.166-7.428-7.067-7.428zm31.466 0c-3.901 0-7.067 3.327-7.067 7.428S97.566 44 101.467 44c3.902 0 7.068-3.328 7.068-7.429 0-4.1-3.166-7.428-7.068-7.428zM14 71.172h324l1.865 198.376c0 2.782-1.651 4.595-4.186 4.595H17.5c-2.535 0-4.187-1.813-4.187-4.595L14 71.172z"/></g><path id="Rectangle-2" stroke="#DBAF88" stroke-width="2" d="M387 7h326v351H387z"/><path id="Rectangle-14" fill="#FFF" d="M395 8h312v61H395z"/><path id="Line-39" fill="#C06334" fill-rule="nonzero" d="M519.5 7v159h6l-7 14-7-14h6V7h2z"/><path id="Line-39-Copy-2" fill="#C06334" fill-rule="nonzero" d="M531.014 141.987l2 .027-.014 1-.311 23 6 .082L531.5 180l-6.81-14.093 5.999.08.311-23 .014-1z"/><text id="pageY" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="525" y="23">pageY</tspan></text><text id="clientY" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="542.4" y="172">clientY</tspan></text><path id="Line-39-Copy" fill="#C06334" fill-rule="nonzero" d="M501 182.5l14 7-14 7-.001-6.001L389 190.5h-1v-2h1l111.999-.001.001-5.999z"/><text id="pageX" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="398" y="183">pageX</tspan></text><text id="clientX" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="16" font-weight="bold"><tspan x="397.4" y="204">clientX</tspan></text><text id="😍" fill="#643B0C" font-family="AppleColorEmoji, Apple Color Emoji" font-size="18" font-weight="normal"><tspan x="514" y="197">😍</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/1-document/11-coordinates/document-window-coordinates-scroll.png b/2-ui/1-document/11-coordinates/document-window-coordinates-scroll.png deleted file mode 100644 index 29c59aad1..000000000 Binary files a/2-ui/1-document/11-coordinates/document-window-coordinates-scroll.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/document-window-coordinates-scroll@2x.png b/2-ui/1-document/11-coordinates/document-window-coordinates-scroll@2x.png deleted file mode 100644 index 3c188ee45..000000000 Binary files a/2-ui/1-document/11-coordinates/document-window-coordinates-scroll@2x.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/document-window-coordinates-zero.png b/2-ui/1-document/11-coordinates/document-window-coordinates-zero.png deleted file mode 100644 index 3dc17f025..000000000 Binary files a/2-ui/1-document/11-coordinates/document-window-coordinates-zero.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/document-window-coordinates-zero@2x.png b/2-ui/1-document/11-coordinates/document-window-coordinates-zero@2x.png deleted file mode 100644 index 45f6b9e7f..000000000 Binary files a/2-ui/1-document/11-coordinates/document-window-coordinates-zero@2x.png and /dev/null differ diff --git a/2-ui/1-document/11-coordinates/head.html b/2-ui/1-document/11-coordinates/head.html index d5b366a9d..c00340039 100644 --- a/2-ui/1-document/11-coordinates/head.html +++ b/2-ui/1-document/11-coordinates/head.html @@ -1,30 +1,30 @@ <script> document.addEventListener('DOMContentLoaded', function() { -let elem = document.getElementById('coords-show-mark'); + let elem = document.getElementById('coords-show-mark'); -// no elem in ebook mode -if (elem) { - elem.onclick = function() { + // no elem in ebook (pdf/epub) mode + if (elem) { + elem.onclick = function() { - function createMessageUnder(elem, text) { - let coords = elem.getBoundingClientRect(); - let message = document.createElement('div'); - message.style.cssText = "position:fixed; color: red"; + function createMessageUnder(elem, text) { + let coords = elem.getBoundingClientRect(); + let message = document.createElement('div'); + message.style.cssText = "position:fixed; color: red"; - message.style.left = coords.left + "px"; - message.style.top = coords.bottom + "px"; + message.style.left = coords.left + "px"; + message.style.top = coords.bottom + "px"; - message.innerHTML = text; + message.innerHTML = text; - return message; - } + return message; + } - let message = createMessageUnder(elem, 'Hello, world!'); - document.body.append(message); - setTimeout(() => message.remove(), 5000); + let message = createMessageUnder(elem, 'Hello, world!'); + document.body.append(message); + setTimeout(() => message.remove(), 5000); + } } -} }); diff --git a/2-ui/2-events/01-introduction-browser-events/02-hide-self-onclick/solution.md b/2-ui/2-events/01-introduction-browser-events/02-hide-self-onclick/solution.md index ec2806696..cded5b622 100644 --- a/2-ui/2-events/01-introduction-browser-events/02-hide-self-onclick/solution.md +++ b/2-ui/2-events/01-introduction-browser-events/02-hide-self-onclick/solution.md @@ -1,4 +1,4 @@ -Can use `this` in the handler to reference "itself" here: +Can use `this` in the handler to reference "the element itself" here: ```html run height=50 <input type="button" onclick="this.hidden=true" value="Click to hide"> diff --git a/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/move-ball-coords.svg b/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/move-ball-coords.svg index fc26b023c..247f75b7b 100644 --- a/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/move-ball-coords.svg +++ b/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/move-ball-coords.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="521" height="330" viewBox="0 0 521 330"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-2" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter><path id="path-1" d="M263 170v21.429l5.25-5.358 4.375 8.929h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H277L263 170z"/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="move-ball-coords.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(12 6)"><path id="Shape" d="M10.438 316C4.676 316 0 312.445 0 308.1V7.9C0 3.555 4.676 0 10.438 0h480.125C496.303 0 501 3.555 501 7.9v300.2c0 4.345-4.697 7.9-10.438 7.9H10.438zm-.276-12h478.643V12.939H10.162V304z"/><path id="Shape" d="M20.859 54.1c-5.753 0-10.422-1.147-10.422-2.55 0-1.402 4.67-2.55 10.422-2.55H479.4c5.732 0 10.422 1.148 10.422 2.55 0 1.403-4.69 2.55-10.422 2.55H20.86zM33.75 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM63.938 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM94.127 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6z"/></g><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="183" y="100" xlink:href=""/><text id="ball.style.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="174" y="262">ball.style.left</tspan></text><text id="?" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="40" font-weight="normal"><tspan x="217.5" y="229">?</tspan></text><text id="fieldCoords.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="48.9" y="189.32">fieldCoords.left</tspan></text><text id="event.clientX" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="59.7" y="161.32">event.clientX</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M168.41 187.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H168.41v-6z"/><path id="Line-41-Copy-2" fill="#C06334" fill-rule="nonzero" d="M248 160.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H248v-6z"/><g id="ball" opacity=".7" transform="translate(261 168)"><circle id="Oval-4" cx="18.753" cy="18.753" r="18.753" fill="#FFF"/><path id="Shape" fill="#181717" fill-rule="nonzero" d="M37.792 18.204a.229.229 0 00-.003-.123l-.004-.016a18.9 18.9 0 00-6.003-12.94c-.445-.461-1.107-1.006-1.863-1.534A19.043 19.043 0 0012.899.935c-.412.106-.98.322-1.64.622a18.593 18.593 0 00-3.53 2.014A18.856 18.856 0 004.68 6.344c-6.874 7.794-6.085 19.762 1.758 26.679a19.043 19.043 0 004.876 3.128c.254.121.465.194.645.272 7.417 2.964 15.919.96 21.169-4.992 3.216-3.65 4.87-8.343 4.663-13.227zm-.677-.785l-.05-.135c-.387-1.04-1.095-2.957-2.828-4.492-.255-3.228-1.405-5.115-2.43-6.774a18.297 18.297 0 015.308 11.401zM24.485 3.158a.476.476 0 00.044-.013c1.666.291 4.245 1.237 6.464 2.748a.274.274 0 00.03.068c.06.1.214.347.214.347.99 1.6 2.109 3.414 2.362 6.494-1.007.741-1.724 1.531-2.42 2.295-.414.456-.84.925-1.326 1.383-1.63-1.516-4.52-2.813-7.641-3.42-.306-1.35-.919-3.854-2.423-6.628l.116-.132c2.418-2.743 3.269-2.9 4.255-3.08.105-.019.213-.039.325-.062zm-8.729 15.517l6.255-5.001c3.137.597 6.025 1.903 7.547 3.413.372 1.153.468 3.636.54 5.464.035.935.069 1.754.127 2.246-2.457 2.24-5.717 4.539-6.78 4.778l-6.843-2.677c-.512-2.867-.662-4.83-.846-8.223zm.205 8.273l-4.955 3.485a.249.249 0 00-.148-.012c-.923.192-4.479-2.415-6.21-3.929a.264.264 0 00-.065-.04c-.073-.495-.23-1.125-.408-1.84-.469-1.885-1.05-4.221-.528-5.673l.028-.028c.006-.007.013-.015.018-.024.339-.487.665-.908 1-1.286 1.058-1.2 2.256-2.053 4.12-2.927l.379.276c1.735 1.266 4.094 2.99 5.927 3.8.185 3.374.335 5.339.842 8.198zM9.35 3.678c.07-.079.145-.147.233-.226l.049-.044c.185-.102.424-.243.702-.405.653-.381 1.546-.905 2.527-1.398a18.483 18.483 0 018.294-.832c0 .093.052.18.137.223.17.089.343.172.515.255.733.354 1.432.692 2.073 1.37-1.001.19-2.025.496-4.464 3.262l-.156.177c-2.457-.245-5.282-.38-7.822.37-.94-1.119-1.73-1.651-2.429-2.123l-.019-.013c.153-.357.266-.51.36-.616zM1.413 15.43a.25.25 0 00-.398-.123 1.375 1.375 0 00-.076.066 18.024 18.024 0 014.08-8.465c.12-.1.274-.233.448-.385.724-.63 2.41-2.101 2.946-1.847a13.536 13.536 0 00.092.063l.147.1c.652.44 1.39.937 2.258 1.96-.78 1.657-2.416 5.752-2.408 7.314-1.927.907-3.177 1.802-4.288 3.062-.295.334-.584.699-.878 1.108l-.106-.1c-.557-.52-1.396-1.306-1.817-2.753zm5.438 17.124a18.323 18.323 0 01-3.95-4.92c.691-.195 1.023-.323 1.411-.586.638.539 4.586 3.827 6.334 4.006a36.692 36.692 0 003.055 4.295c-.572.352-1.042.473-1.648.437a18.311 18.311 0 01-5.202-3.232zm6.469 3.698c.295-.108.593-.268.92-.49 2.756.407 5.828.152 7.99-.152.031.021.066.034.105.04.52.068 2.062.226 3.92.015a18.292 18.292 0 01-12.935.587zm18.778-4.768a.256.256 0 00-.042.181 17.964 17.964 0 01-3.83 3.004.236.236 0 00-.111.003c-2.244.59-4.392.479-5.354.382.433-1.078.73-2.68.902-4.876 1.262-.352 4.456-2.594 7.01-4.925.636.105 1.07.204 1.421.286.858.198 1.25.29 2.81.086-.201 1.17-.765 2.896-2.806 5.86zm2.797-3.623c.438-1.046.605-1.827.69-2.448a.237.237 0 00-.014-.125c.39-.886 1.186-3.392 1.594-5.443a18.028 18.028 0 01-2.27 8.016z"/></g><g id="Default"><use fill="#000" filter="url(#filter-2)" xlink:href="#path-1"/><path fill="#FFF" stroke="#A7333A" d="M262.5 168.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727v-23.878z"/></g><path id="Line-16-Copy-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M195.535 180.945s20.728 10.637 32.233 10.637c11.504 0 32.232-10.637 32.232-10.637"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="521" height="330" viewBox="0 0 521 330"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-2" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter><path id="path-1" d="M263 170v21.429l5.25-5.358 4.375 8.929h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H277L263 170z"/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="move-ball-coords.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(12 6)"><path id="Shape" d="M10.438 316C4.676 316 0 312.445 0 308.1V7.9C0 3.555 4.676 0 10.438 0h480.125C496.303 0 501 3.555 501 7.9v300.2c0 4.345-4.697 7.9-10.438 7.9H10.438zm-.276-12h478.643V12.939H10.162V304z"/><path id="Shape" d="M20.859 54.1c-5.753 0-10.422-1.147-10.422-2.55 0-1.402 4.67-2.55 10.422-2.55H479.4c5.732 0 10.422 1.148 10.422 2.55 0 1.403-4.69 2.55-10.422 2.55H20.86zM33.75 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM63.938 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM94.127 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6z"/></g><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="183" y="100" xlink:href=""/><text id="ball.style.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="174" y="262">ball.style.left</tspan></text><text id="?" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="40" font-weight="normal"><tspan x="217.5" y="229">?</tspan></text><text id="fieldCoords.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="48.9" y="189.32">fieldCoords.left</tspan></text><text id="event.clientX" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="59.7" y="161.32">event.clientX</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M168.41 187.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H168.41v-6z"/><path id="Line-41-Copy-2" fill="#C06334" fill-rule="nonzero" d="M248 160.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H248v-6z"/><g id="ball" opacity=".7" transform="translate(261 168)"><circle id="Oval-4" cx="18.753" cy="18.753" r="18.753" fill="#FFF"/><path id="Shape" fill="#181717" fill-rule="nonzero" d="M37.792 18.204a.229.229 0 00-.003-.123l-.004-.016a18.9 18.9 0 00-6.003-12.94c-.445-.461-1.107-1.006-1.863-1.534A19.043 19.043 0 0012.899.935c-.412.106-.98.322-1.64.622a18.593 18.593 0 00-3.53 2.014A18.856 18.856 0 004.68 6.344c-6.874 7.794-6.085 19.762 1.758 26.679a19.043 19.043 0 004.876 3.128c.254.121.465.194.645.272 7.417 2.964 15.919.96 21.169-4.992 3.216-3.65 4.87-8.343 4.663-13.227zm-.677-.785l-.05-.135c-.387-1.04-1.095-2.957-2.828-4.492-.255-3.228-1.405-5.115-2.43-6.774a18.297 18.297 0 015.308 11.401zM24.485 3.158a.476.476 0 00.044-.013c1.666.291 4.245 1.237 6.464 2.748a.274.274 0 00.03.068c.06.1.214.347.214.347.99 1.6 2.109 3.414 2.362 6.494-1.007.741-1.724 1.531-2.42 2.295-.414.456-.84.925-1.326 1.383-1.63-1.516-4.52-2.813-7.641-3.42-.306-1.35-.919-3.854-2.423-6.628l.116-.132c2.418-2.743 3.269-2.9 4.255-3.08.105-.019.213-.039.325-.062zm-8.729 15.517l6.255-5.001c3.137.597 6.025 1.903 7.547 3.413.372 1.153.468 3.636.54 5.464.035.935.069 1.754.127 2.246-2.457 2.24-5.717 4.539-6.78 4.778l-6.843-2.677c-.512-2.867-.662-4.83-.846-8.223zm.205 8.273l-4.955 3.485a.249.249 0 00-.148-.012c-.923.192-4.479-2.415-6.21-3.929a.264.264 0 00-.065-.04c-.073-.495-.23-1.125-.408-1.84-.469-1.885-1.05-4.221-.528-5.673l.028-.028c.006-.007.013-.015.018-.024.339-.487.665-.908 1-1.286 1.058-1.2 2.256-2.053 4.12-2.927l.379.276c1.735 1.266 4.094 2.99 5.927 3.8.185 3.374.335 5.339.842 8.198zM9.35 3.678c.07-.079.145-.147.233-.226l.049-.044c.185-.102.424-.243.702-.405.653-.381 1.546-.905 2.527-1.398a18.483 18.483 0 018.294-.832c0 .093.052.18.137.223.17.089.343.172.515.255.733.354 1.432.692 2.073 1.37-1.001.19-2.025.496-4.464 3.262l-.156.177c-2.457-.245-5.282-.38-7.822.37-.94-1.119-1.73-1.651-2.429-2.123l-.019-.013c.153-.357.266-.51.36-.616zM1.413 15.43a.25.25 0 00-.398-.123 1.375 1.375 0 00-.076.066 18.024 18.024 0 014.08-8.465c.12-.1.274-.233.448-.385.724-.63 2.41-2.101 2.946-1.847a13.536 13.536 0 00.092.063l.147.1c.652.44 1.39.937 2.258 1.96-.78 1.657-2.416 5.752-2.408 7.314-1.927.907-3.177 1.802-4.288 3.062-.295.334-.584.699-.878 1.108l-.106-.1c-.557-.52-1.396-1.306-1.817-2.753zm5.438 17.124a18.323 18.323 0 01-3.95-4.92c.691-.195 1.023-.323 1.411-.586.638.539 4.586 3.827 6.334 4.006a36.692 36.692 0 003.055 4.295c-.572.352-1.042.473-1.648.437a18.311 18.311 0 01-5.202-3.232zm6.469 3.698c.295-.108.593-.268.92-.49 2.756.407 5.828.152 7.99-.152.031.021.066.034.105.04.52.068 2.062.226 3.92.015a18.292 18.292 0 01-12.935.587zm18.778-4.768a.256.256 0 00-.042.181 17.964 17.964 0 01-3.83 3.004.236.236 0 00-.111.003c-2.244.59-4.392.479-5.354.382.433-1.078.73-2.68.902-4.876 1.262-.352 4.456-2.594 7.01-4.925.636.105 1.07.204 1.421.286.858.198 1.25.29 2.81.086-.201 1.17-.765 2.896-2.806 5.86zm2.797-3.623c.438-1.046.605-1.827.69-2.448a.237.237 0 00-.014-.125c.39-.886 1.186-3.392 1.594-5.443a18.028 18.028 0 01-2.27 8.016z"/></g><g id="Default"><use fill="#000" filter="url(#filter-2)" xlink:href="#path-1"/><path fill="#FFF" stroke="#A7333A" d="M262.5 168.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727v-23.878z"/></g><path id="Line-16-Copy-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M195.535 180.945s20.728 10.637 32.233 10.637c11.504 0 32.232-10.637 32.232-10.637"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="521" height="330" viewBox="0 0 521 330"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-2" width="195.6%" height="153.8%" x="-42.9%" y="-25.8%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter><path id="path-1" d="M263 170v21.429l5.25-5.358 4.375 8.929h1.75s1.13-1.161.875-1.786c-1.203-2.947-4.375-8.928-4.375-8.928H277L263 170z"/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="move-ball-coords.svg"><g id="noun_69008_cc" fill="#DBAF88" transform="translate(12 6)"><path id="Shape" d="M10.438 316C4.676 316 0 312.445 0 308.1V7.9C0 3.555 4.676 0 10.438 0h480.125C496.303 0 501 3.555 501 7.9v300.2c0 4.345-4.697 7.9-10.438 7.9H10.438zm-.276-12h478.643V12.939H10.162V304z"/><path id="Shape" d="M20.859 54.1c-5.753 0-10.422-1.147-10.422-2.55 0-1.402 4.67-2.55 10.422-2.55H479.4c5.732 0 10.422 1.148 10.422 2.55 0 1.403-4.69 2.55-10.422 2.55H20.86zM33.75 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM63.938 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6zM94.127 31.6c0 3.315-2.685 6-6 6s-6-2.685-6-6 2.685-6 6-6 6 2.685 6 6z"/></g><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="183" y="100" xlink:href=""/><text id="ball.style.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="174" y="262">ball.style.left</tspan></text><text id="?" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="40" font-weight="normal"><tspan x="217.5" y="229">?</tspan></text><text id="fieldCoords.left" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="48.9" y="189.32">fieldCoords.left</tspan></text><text id="event.clientX" fill="#C06334" font-family="PTMono-Regular, PT Mono" font-size="12" font-weight="normal"><tspan x="59.7" y="161.32">event.clientX</tspan></text><path id="Line-41" fill="#C06334" fill-rule="nonzero" d="M168.41 187.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H168.41v-6z"/><path id="Line-41-Copy-2" fill="#C06334" fill-rule="nonzero" d="M248 160.91l14 7-14 7v-6H39.589l.001 6-14-7 14-7-.001 6H248v-6z"/><g id="ball" opacity=".7" transform="translate(261 168)"><circle id="Oval-4" cx="18.753" cy="18.753" r="18.753" fill="#FFF"/><path id="Shape" fill="#181717" fill-rule="nonzero" d="M37.792 18.204a.229.229 0 00-.003-.123l-.004-.016a18.9 18.9 0 00-6.003-12.94c-.445-.461-1.107-1.006-1.863-1.534A19.043 19.043 0 0012.899.935c-.412.106-.98.322-1.64.622a18.593 18.593 0 00-3.53 2.014A18.856 18.856 0 004.68 6.344c-6.874 7.794-6.085 19.762 1.758 26.679a19.043 19.043 0 004.876 3.128c.254.121.465.194.645.272 7.417 2.964 15.919.96 21.169-4.992 3.216-3.65 4.87-8.343 4.663-13.227zm-.677-.785l-.05-.135c-.387-1.04-1.095-2.957-2.828-4.492-.255-3.228-1.405-5.115-2.43-6.774a18.297 18.297 0 015.308 11.401zM24.485 3.158a.476.476 0 00.044-.013c1.666.291 4.245 1.237 6.464 2.748a.274.274 0 00.03.068c.06.1.214.347.214.347.99 1.6 2.109 3.414 2.362 6.494-1.007.741-1.724 1.531-2.42 2.295-.414.456-.84.925-1.326 1.383-1.63-1.516-4.52-2.813-7.641-3.42-.306-1.35-.919-3.854-2.423-6.628l.116-.132c2.418-2.743 3.269-2.9 4.255-3.08.105-.019.213-.039.325-.062zm-8.729 15.517l6.255-5.001c3.137.597 6.025 1.903 7.547 3.413.372 1.153.468 3.636.54 5.464.035.935.069 1.754.127 2.246-2.457 2.24-5.717 4.539-6.78 4.778l-6.843-2.677c-.512-2.867-.662-4.83-.846-8.223zm.205 8.273l-4.955 3.485a.249.249 0 00-.148-.012c-.923.192-4.479-2.415-6.21-3.929a.264.264 0 00-.065-.04c-.073-.495-.23-1.125-.408-1.84-.469-1.885-1.05-4.221-.528-5.673l.028-.028c.006-.007.013-.015.018-.024.339-.487.665-.908 1-1.286 1.058-1.2 2.256-2.053 4.12-2.927l.379.276c1.735 1.266 4.094 2.99 5.927 3.8.185 3.374.335 5.339.842 8.198zM9.35 3.678c.07-.079.145-.147.233-.226l.049-.044c.185-.102.424-.243.702-.405.653-.381 1.546-.905 2.527-1.398a18.483 18.483 0 018.294-.832c0 .093.052.18.137.223.17.089.343.172.515.255.733.354 1.432.692 2.073 1.37-1.001.19-2.025.496-4.464 3.262l-.156.177c-2.457-.245-5.282-.38-7.822.37-.94-1.119-1.73-1.651-2.429-2.123l-.019-.013c.153-.357.266-.51.36-.616zM1.413 15.43a.25.25 0 00-.398-.123 1.375 1.375 0 00-.076.066 18.024 18.024 0 014.08-8.465c.12-.1.274-.233.448-.385.724-.63 2.41-2.101 2.946-1.847a13.536 13.536 0 00.092.063l.147.1c.652.44 1.39.937 2.258 1.96-.78 1.657-2.416 5.752-2.408 7.314-1.927.907-3.177 1.802-4.288 3.062-.295.334-.584.699-.878 1.108l-.106-.1c-.557-.52-1.396-1.306-1.817-2.753zm5.438 17.124a18.323 18.323 0 01-3.95-4.92c.691-.195 1.023-.323 1.411-.586.638.539 4.586 3.827 6.334 4.006a36.692 36.692 0 003.055 4.295c-.572.352-1.042.473-1.648.437a18.311 18.311 0 01-5.202-3.232zm6.469 3.698c.295-.108.593-.268.92-.49 2.756.407 5.828.152 7.99-.152.031.021.066.034.105.04.52.068 2.062.226 3.92.015a18.292 18.292 0 01-12.935.587zm18.778-4.768a.256.256 0 00-.042.181 17.964 17.964 0 01-3.83 3.004.236.236 0 00-.111.003c-2.244.59-4.392.479-5.354.382.433-1.078.73-2.68.902-4.876 1.262-.352 4.456-2.594 7.01-4.925.636.105 1.07.204 1.421.286.858.198 1.25.29 2.81.086-.201 1.17-.765 2.896-2.806 5.86zm2.797-3.623c.438-1.046.605-1.827.69-2.448a.237.237 0 00-.014-.125c.39-.886 1.186-3.392 1.594-5.443a18.028 18.028 0 01-2.27 8.016z"/></g><g id="Default"><use fill="#000" filter="url(#filter-2)" xlink:href="#path-1"/><path fill="#FFF" stroke="#A7333A" d="M262.5 168.775l15.69 16.01h-6.488c.886 1.695 3.06 5.91 4.01 8.24.318.776-.979 2.324-.979 2.324h0l-2.42.151-4.2-8.574-5.613 5.727v-23.878z"/></g><path id="Line-16-Copy-2" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M195.535 180.945s20.728 10.637 32.233 10.637c11.504 0 32.232-10.637 32.232-10.637"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/solution.md b/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/solution.md index 6268b2aa8..b04cb8231 100644 --- a/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/solution.md +++ b/2-ui/2-events/01-introduction-browser-events/04-move-ball-field/solution.md @@ -22,7 +22,7 @@ Then the ball will be positioned relatively to the field: } ``` -Next we need to assign the correct `ball.style.position.left/top`. They contain field-relative coordinates now. +Next we need to assign the correct `ball.style.left/top`. They contain field-relative coordinates now. Here's the picture: @@ -36,7 +36,7 @@ To get field-relative `left` coordinate of the click, we can substract the field let left = event.clientX - fieldCoords.left - field.clientLeft; ``` -Normally, `ball.style.position.left` means the "left edge of the element" (the ball). So if we assign that `left`, then the ball edge would be under the mouse cursor. +Normally, `ball.style.left` means the "left edge of the element" (the ball). So if we assign that `left`, then the ball edge, not center, would be under the mouse cursor. We need to move the ball half-width left and half-height up to make it center. diff --git a/2-ui/2-events/01-introduction-browser-events/05-sliding-menu/solution.md b/2-ui/2-events/01-introduction-browser-events/05-sliding-menu/solution.md index eed096231..7554a2f09 100644 --- a/2-ui/2-events/01-introduction-browser-events/05-sliding-menu/solution.md +++ b/2-ui/2-events/01-introduction-browser-events/05-sliding-menu/solution.md @@ -29,7 +29,7 @@ Like this: So if we set `onclick` on it, then it will catch clicks to the right of the text. -...but `<span>` has an implicit `display: inline`, so it occupies exactly enough place to fit all the text: +As `<span>` has an implicit `display: inline`, it occupies exactly enough place to fit all the text: ```html autorun height=50 <span style="border: solid red 1px" onclick="alert(1)">Sweeties (click me)!</span> diff --git a/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.md b/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.md index 41de7e1f3..1c6b52cea 100644 --- a/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.md +++ b/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.md @@ -1,6 +1,6 @@ The images ribbon can be represented as `ul/li` list of images `<img>`. -Normally, such a ribbon is wide, but we put a fixed-size `<div>` around to "cut" it, so that only a part of the ribbon is visibble: +Normally, such a ribbon is wide, but we put a fixed-size `<div>` around to "cut" it, so that only a part of the ribbon is visible:  diff --git a/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.view/index.html b/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.view/index.html index 2c6073316..baf867664 100644 --- a/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.view/index.html +++ b/2-ui/2-events/01-introduction-browser-events/07-carousel/solution.view/index.html @@ -10,7 +10,7 @@ <div id="carousel" class="carousel"> <button class="arrow prev">⇦</button> <div class="gallery"> - <ul class="images"> + <ul> <li><img src="https://en.js.cx/carousel/1.png"></li> <li><img src="https://en.js.cx/carousel/2.png"></li> <li><img src="https://en.js.cx/carousel/3.png"></li> diff --git a/2-ui/2-events/01-introduction-browser-events/article.md b/2-ui/2-events/01-introduction-browser-events/article.md index 332d15793..4eca222aa 100644 --- a/2-ui/2-events/01-introduction-browser-events/article.md +++ b/2-ui/2-events/01-introduction-browser-events/article.md @@ -11,20 +11,20 @@ Here's a list of the most useful DOM events, just to take a look at: - `mousedown` / `mouseup` -- when the mouse button is pressed / released over an element. - `mousemove` -- when the mouse is moved. +**Keyboard events:** +- `keydown` and `keyup` -- when a keyboard key is pressed and released. + **Form element events:** - `submit` -- when the visitor submits a `<form>`. - `focus` -- when the visitor focuses on an element, e.g. on an `<input>`. -**Keyboard events:** -- `keydown` and `keyup` -- when the visitor presses and then releases the button. - **Document events:** - `DOMContentLoaded` -- when the HTML is loaded and processed, DOM is fully built. **CSS events:** - `transitionend` -- when a CSS-animation finishes. -There are many other events. We'll get into more details of particular events in next chapters. +There are many other events. We'll get into more details of particular events in upcoming chapters. ## Event handlers @@ -87,8 +87,6 @@ If the handler is assigned using an HTML-attribute then the browser reads it, cr So this way is actually the same as the previous one. -**The handler is always in the DOM property: the HTML-attribute is just one of the ways to initialize it.** - These two code pieces work the same: 1. Only HTML: @@ -109,6 +107,8 @@ These two code pieces work the same: </script> ``` +In the first example, the HTML attribute is used to initialize the `button.onclick`, while in the second example -- the script, that's all the difference. + **As there's only one `onclick` property, we can't assign more than one event handler.** In the example below adding a handler with JavaScript overwrites the existing handler: @@ -124,16 +124,6 @@ In the example below adding a handler with JavaScript overwrites the existing ha </script> ``` -By the way, we can assign an existing function as a handler directly: - -```js -function sayThanks() { - alert('Thanks!'); -} - -elem.onclick = sayThanks; -``` - To remove a handler -- assign `elem.onclick = null`. ## Accessing the element: this @@ -148,9 +138,19 @@ In the code below `button` shows its contents using `this.innerHTML`: ## Possible mistakes -If you're starting to work with event -- please note some subtleties. +If you're starting to work with events -- please note some subtleties. + +We can set an existing function as a handler: + +```js +function sayThanks() { + alert('Thanks!'); +} + +elem.onclick = sayThanks; +``` -**The function should be assigned as `sayThanks`, not `sayThanks()`.** +But be careful: the function should be assigned as `sayThanks`, not `sayThanks()`. ```js // right @@ -160,29 +160,25 @@ button.onclick = sayThanks; button.onclick = sayThanks(); ``` -If we add brackets, then `sayThanks()` -- will be the *result* of the function execution, so `onclick` in the last code becomes `undefined` (the function returns nothing). That won't work. +If we add parentheses, then `sayThanks()` becomes a function call. So the last line actually takes the *result* of the function execution, that is `undefined` (as the function returns nothing), and assigns it to `onclick`. That doesn't work. -...But in the markup we do need the brackets: +...On the other hand, in the markup we do need the parentheses: ```html <input type="button" id="button" onclick="sayThanks()"> ``` -The difference is easy to explain. When the browser reads the attribute, it creates a handler function with the body from its content. +The difference is easy to explain. When the browser reads the attribute, it creates a handler function with body from the attribute content. -So the last example is the same as: +So the markup generates this property: ```js button.onclick = function() { *!* - sayThanks(); // the attribute content + sayThanks(); // <-- the attribute content goes here */!* }; ``` -**Use functions, not strings.** - -The assignment `elem.onclick = "alert(1)"` would work too. It works for compatibility reasons, but strongly not recommended. - **Don't use `setAttribute` for handlers.** Such a call won't work: @@ -199,9 +195,9 @@ Assign a handler to `elem.onclick`, not `elem.ONCLICK`, because DOM properties a ## addEventListener -The fundamental problem of the aforementioned ways to assign handlers -- we can't assign multiple handlers to one event. +The fundamental problem of the aforementioned ways to assign handlers is that we *can't assign multiple handlers to one event*. -For instance, one part of our code wants to highlight a button on click, and another one wants to show a message. +Let's say, one part of our code wants to highlight a button on click, and another one wants to show a message on the same click. We'd like to assign two event handlers for that. But a new DOM property will overwrite the existing one: @@ -211,12 +207,12 @@ input.onclick = function() { alert(1); } input.onclick = function() { alert(2); } // replaces the previous handler ``` -Web-standard developers understood that long ago and suggested an alternative way of managing handlers using special methods `addEventListener` and `removeEventListener`. They are free of such a problem. +Developers of web standards understood that long ago and suggested an alternative way of managing handlers using the special methods `addEventListener` and `removeEventListener` which aren't bound by such constraint. The syntax to add a handler: ```js -element.addEventListener(event, handler[, options]); +element.addEventListener(event, handler, [options]); ``` `event` @@ -228,20 +224,19 @@ element.addEventListener(event, handler[, options]); `options` : An additional optional object with properties: - `once`: if `true`, then the listener is automatically removed after it triggers. - - `capture`: the phrase where to handle the event, to be covered later in the chapter <info:bubbling-and-capturing>. For historical reasons, `options` can also be `false/true`, that's the same as `{capture: false/true}`. - - `passive`: if `true`, then the handler will not `preventDefault()`, we'll cover that later in <info:default-browser-action>. - + - `capture`: the phase where to handle the event, to be covered later in the chapter <info:bubbling-and-capturing>. For historical reasons, `options` can also be `false/true`, that's the same as `{capture: false/true}`. + - `passive`: if `true`, then the handler will not call `preventDefault()`, we'll explain that later in <info:default-browser-action>. To remove the handler, use `removeEventListener`: ```js -element.removeEventListener(event, handler[, options]); +element.removeEventListener(event, handler, [options]); ``` ````warn header="Removal requires the same function" To remove a handler we should pass exactly the same function as was assigned. -That doesn't work: +This doesn't work: ```js no-beautify elem.addEventListener( "click" , () => alert('Thanks!')); @@ -249,7 +244,7 @@ elem.addEventListener( "click" , () => alert('Thanks!')); elem.removeEventListener( "click", () => alert('Thanks!')); ``` -The handler won't be removed, because `removeEventListener` gets another function -- with the same code, but that doesn't matter. +The handler won't be removed, because `removeEventListener` gets another function -- with the same code, but that doesn't matter, as it's a different function object. Here's the right way: @@ -266,7 +261,7 @@ input.removeEventListener("click", handler); Please note -- if we don't store the function in a variable, then we can't remove it. There's no way to "read back" handlers assigned by `addEventListener`. ```` -Multiple calls to `addEventListener` allow to add multiple handlers, like this: +Multiple calls to `addEventListener` allow it to add multiple handlers, like this: ```html run no-beautify <input id="elem" type="button" value="Click me"/> @@ -291,47 +286,33 @@ Multiple calls to `addEventListener` allow to add multiple handlers, like this: As we can see in the example above, we can set handlers *both* using a DOM-property and `addEventListener`. But generally we use only one of these ways. ````warn header="For some events, handlers only work with `addEventListener`" -There exist events that can't be assigned via a DOM-property. Must use `addEventListener`. - -For instance, the event `transitionend` (CSS animation finished) is like that. - -Try the code below. In most browsers only the second handler works, not the first one. +There exist events that can't be assigned via a DOM-property. Only with `addEventListener`. -```html run -<style> - input { - transition: width 1s; - width: 100px; - } - - .wide { - width: 300px; - } -</style> - -<input type="button" id="elem" onclick="this.classList.toggle('wide')" value="Click me"> +For instance, the `DOMContentLoaded` event, that triggers when the document is loaded and the DOM has been built. -<script> - elem.ontransitionend = function() { - alert("DOM property"); // doesn't work - }; +```js +// will never run +document.onDOMContentLoaded = function() { + alert("DOM built"); +}; +``` -*!* - elem.addEventListener("transitionend", function() { - alert("addEventListener"); // shows up when the animation finishes - }); -*/!* -</script> +```js +// this way it works +document.addEventListener("DOMContentLoaded", function() { + alert("DOM built"); +}); ``` +So `addEventListener` is more universal. Although, such events are an exception rather than the rule. ```` ## Event object -To properly handle an event we'd want to know more about what's happened. Not just a "click" or a "keypress", but what were the pointer coordinates? Which key was pressed? And so on. +To properly handle an event we'd want to know more about what's happened. Not just a "click" or a "keydown", but what were the pointer coordinates? Which key was pressed? And so on. When an event happens, the browser creates an *event object*, puts details into it and passes it as an argument to the handler. -Here's an example of getting mouse coordinates from the event object: +Here's an example of getting pointer coordinates from the event object: ```html run <input type="button" value="Click me" id="elem"> @@ -351,14 +332,14 @@ Some properties of `event` object: : Event type, here it's `"click"`. `event.currentTarget` -: Element that handled the event. That's exactly the same as `this`, unless you bind `this` to something else, and then `event.currentTarget` becomes useful. +: Element that handled the event. That's exactly the same as `this`, unless the handler is an arrow function, or its `this` is bound to something else, then we can get the element from `event.currentTarget`. -`event.clientX / event.clientY` -: Window-relative coordinates of the cursor, for mouse events. +`event.clientX` / `event.clientY` +: Window-relative coordinates of the cursor, for pointer events. -There are more properties. They depend on the event type, so we'll study them later when we come to different events in details. +There are more properties. Many of them depend on the event type: keyboard events have one set of properties, pointer events - another one, we'll study them later when as we move on to the details of different events. -````smart header="The event object is also accessible from HTML" +````smart header="The event object is also available in HTML handlers" If we assign a handler in HTML, we can also use the `event` object, like this: ```html autorun height=60 @@ -371,7 +352,7 @@ That's possible because when the browser reads the attribute, it creates a handl ## Object handlers: handleEvent -We can assign an object as an event handler using `addEventListener`. When an event occurs, its `handleEvent` method is called with it. +We can assign not just a function, but an object as an event handler using `addEventListener`. When an event occurs, its `handleEvent` method is called. For instance: @@ -380,17 +361,19 @@ For instance: <button id="elem">Click me</button> <script> - elem.addEventListener('click', { + let obj = { handleEvent(event) { alert(event.type + " at " + event.currentTarget); } - }); + }; + + elem.addEventListener('click', obj); </script> ``` -In other words, when `addEventListener` receives an object as the handler, it calls `object.handleEvent(event)` in case of an event. +As we can see, when `addEventListener` receives an object as the handler, it calls `obj.handleEvent(event)` in case of an event. -We could also use a class for that: +We could also use objects of a custom class, like this: ```html run @@ -412,6 +395,7 @@ We could also use a class for that: *!* let menu = new Menu(); + elem.addEventListener('mousedown', menu); elem.addEventListener('mouseup', menu); */!* @@ -462,7 +446,7 @@ HTML attributes are used sparingly, because JavaScript in the middle of an HTML DOM properties are ok to use, but we can't assign more than one handler of the particular event. In many cases that limitation is not pressing. -The last way is the most flexible, but it is also the longest to write. There are few events that only work with it, for instance `transtionend` and `DOMContentLoaded` (to be covered). Also `addEventListener` supports objects as event handlers. In that case the method `handleEvent` is called in case of the event. +The last way is the most flexible, but it is also the longest to write. There are few events that only work with it, for instance `transitionend` and `DOMContentLoaded` (to be covered). Also `addEventListener` supports objects as event handlers. In that case the method `handleEvent` is called in case of the event. No matter how you assign the handler -- it gets an event object as the first argument. That object contains the details about what's happened. diff --git a/2-ui/2-events/02-bubbling-and-capturing/article.md b/2-ui/2-events/02-bubbling-and-capturing/article.md index 7269d9ba8..2448cfa5b 100644 --- a/2-ui/2-events/02-bubbling-and-capturing/article.md +++ b/2-ui/2-events/02-bubbling-and-capturing/article.md @@ -68,14 +68,14 @@ For instance, if we have a single handler `form.onclick`, then it can "catch" al In `form.onclick` handler: -- `this` (`=event.currentTarget`) is the `<form>` element, because the handler runs on it. -- `event.target` is the concrete element inside the form that actually was clicked. +- `this` (=`event.currentTarget`) is the `<form>` element, because the handler runs on it. +- `event.target` is the actual element inside the form that was clicked. Check it out: [codetabs height=220 src="bubble-target"] -It's possible that `event.target` equals `this` -- when the click is made directly on the `<form>` element. +It's possible that `event.target` could equal `this` -- it happens when the click is made directly on the `<form>` element. ## Stopping bubbling @@ -102,7 +102,7 @@ To stop the bubbling and prevent handlers on the current element from running, t ``` ```warn header="Don't stop bubbling without a need!" -Bubbling is convenient. Don't stop it without a real need: obvious and architecturally well-thought. +Bubbling is convenient. Don't stop it without a real need: obvious and architecturally well thought out. Sometimes `event.stopPropagation()` creates hidden pitfalls that later may become problems. @@ -110,7 +110,7 @@ For instance: 1. We create a nested menu. Each submenu handles clicks on its elements and calls `stopPropagation` so that the outer menu won't trigger. 2. Later we decide to catch clicks on the whole window, to track users' behavior (where people click). Some analytic systems do that. Usually the code uses `document.addEventListener('click'…)` to catch all clicks. -3. Our analytic won't work over the area where clicks are stopped by `stopPropagation`. We've got a "dead zone". +3. Our analytic won't work over the area where clicks are stopped by `stopPropagation`. Sadly, we've got a "dead zone". There's usually no real need to prevent the bubbling. A task that seemingly requires that may be solved by other means. One of them is to use custom events, we'll cover them later. Also we can write our data into the `event` object in one handler and read it in another one, so we can pass to handlers on parents information about the processing below. ``` @@ -120,26 +120,27 @@ There's usually no real need to prevent the bubbling. A task that seemingly requ There's another phase of event processing called "capturing". It is rarely used in real code, but sometimes can be useful. -The standard [DOM Events](http://www.w3.org/TR/DOM-Level-3-Events/) describes 3 phases of event propagation: +The standard [DOM Events](https://www.w3.org/TR/DOM-Level-3-Events/) describes 3 phases of event propagation: 1. Capturing phase -- the event goes down to the element. 2. Target phase -- the event reached the target element. 3. Bubbling phase -- the event bubbles up from the element. -Here's the picture of a click on `<td>` inside a table, taken from the specification: +Here's the picture, taken from the specification, of the capturing `(1)`, target `(2)` and bubbling `(3)` phases for a click event on a `<td>` inside a table:  -That is: for a click on `<td>` the event first goes through the ancestors chain down to the element (capturing), then it reaches the target, and then it goes up (bubbles), calling handlers on its way. +That is: for a click on `<td>` the event first goes through the ancestors chain down to the element (capturing phase), then it reaches the target and triggers there (target phase), and then it goes up (bubbling phase), calling handlers on its way. -**Before we only talked about bubbling, because the capturing phase is rarely used. Normally it is invisible to us.** +Until now, we only talked about bubbling, because the capturing phase is rarely used. -Handlers added using `on<event>`-property or using HTML attributes or using `addEventListener(event, handler)` don't know anything about capturing, they only run on the 2nd and 3rd phases. +In fact, the capturing phase was invisible for us, because handlers added using `on<event>`-property or using HTML attributes or using two-argument `addEventListener(event, handler)` don't know anything about capturing, they only run on the 2nd and 3rd phases. To catch an event on the capturing phase, we need to set the handler `capture` option to `true`: ```js elem.addEventListener(..., {capture: true}) + // or, just "true" is an alias to {capture: true} elem.addEventListener(..., true) ``` @@ -149,11 +150,10 @@ There are two possible values of the `capture` option: - If it's `false` (default), then the handler is set on the bubbling phase. - If it's `true`, then the handler is set on the capturing phase. -Note that while formally there are 3 phases, the 2nd phase ("target phase": the event reached the element) is not handled separately: handlers on both capturing and bubbling phases trigger at that phase. -If one puts capturing and bubbling handlers on the target element, the capture handler triggers last in the capturing phase and the bubble handler triggers first in the bubbling phase. +Note that while formally there are 3 phases, the 2nd phase ("target phase": the event reached the element) is not handled separately: handlers on both capturing and bubbling phases trigger at that phase. -Let's see it in action: +Let's see both capturing and bubbling in action: ```html run autorun height=140 edit <style> @@ -181,10 +181,10 @@ The code sets click handlers on *every* element in the document to see which one If you click on `<p>`, then the sequence is: -1. `HTML` -> `BODY` -> `FORM` -> `DIV` -> `P` (capturing phase, the first listener), and then: +1. `HTML` -> `BODY` -> `FORM` -> `DIV -> P` (capturing phase, the first listener): 2. `P` -> `DIV` -> `FORM` -> `BODY` -> `HTML` (bubbling phase, the second listener). -Please note that `P` shows up two times: at the end of capturing and at the start of bubbling. +Please note, the `P` shows up twice, because we've set two listeners: capturing and bubbling. The target triggers at the end of the first and at the beginning of the second phase. There's a property `event.eventPhase` that tells us the number of the phase on which the event was caught. But it's rarely used, because we usually know it in the handler. @@ -192,26 +192,42 @@ There's a property `event.eventPhase` that tells us the number of the phase on w If we `addEventListener(..., true)`, then we should mention the same phase in `removeEventListener(..., true)` to correctly remove the handler. ``` +````smart header="Listeners on the same element and same phase run in their set order" +If we have multiple event handlers on the same phase, assigned to the same element with `addEventListener`, they run in the same order as they are created: + +```js +elem.addEventListener("click", e => alert(1)); // guaranteed to trigger first +elem.addEventListener("click", e => alert(2)); +``` +```` + +```smart header="The `event.stopPropagation()` during the capturing also prevents the bubbling" +The `event.stopPropagation()` method and its sibling `event.stopImmediatePropagation()` can also be called on the capturing phase. Then not only the futher capturing is stopped, but the bubbling as well. + +In other words, normally the event goes first down ("capturing") and then up ("bubbling"). But if `event.stopPropagation()` is called during the capturing phase, then the event travel stops, no bubbling will occur. +``` + + ## Summary -The event handling process: +When an event happens -- the most nested element where it happens gets labeled as the "target element" (`event.target`). -- When an event happens -- the most nested element where it happens gets labeled as the "target element" (`event.target`). -- Then the event first moves from the document root down to the `event.target`, calling handlers assigned with `addEventListener(...., true)` on the way (`true` is a shorthand for `{capture: true}`). -- Then the event moves from `event.target` up to the root, calling handlers assigned using `on<event>` and `addEventListener` without the 3rd argument or with the 3rd argument `false`. +- Then the event moves down from the document root to `event.target`, calling handlers assigned with `addEventListener(..., true)` on the way (`true` is a shorthand for `{capture: true}`). +- Then handlers are called on the target element itself. +- Then the event bubbles up from `event.target` to the root, calling handlers assigned using `on<event>`, HTML attributes and `addEventListener` without the 3rd argument or with the 3rd argument `false/{capture:false}`. Each handler can access `event` object properties: - `event.target` -- the deepest element that originated the event. - `event.currentTarget` (=`this`) -- the current element that handles the event (the one that has the handler on it) -- `event.eventPhase` -- the current phase (capturing=1, bubbling=3). +- `event.eventPhase` -- the current phase (capturing=1, target=2, bubbling=3). Any event handler can stop the event by calling `event.stopPropagation()`, but that's not recommended, because we can't really be sure we won't need it above, maybe for completely different things. -The capturing phase is used very rarely, usually we handle events on bubbling. And there's a logic behind that. +The capturing phase is used very rarely, usually we handle events on bubbling. And there's a logical explanation for that. In real world, when an accident happens, local authorities react first. They know best the area where it happened. Then higher-level authorities if needed. -The same for event handlers. The code that set the handler on a particular element knows maximum details about the element and what it does. A handler on a particular `<td>` may be suited for that exactly `<td>`, it knows everything about it, so it should get the chance first. Then its immediate parent also knows about the context, but a little bit less, and so on till the very top element that handles general concepts and runs the last. +The same for event handlers. The code that set the handler on a particular element knows maximum details about the element and what it does. A handler on a particular `<td>` may be suited for that exactly `<td>`, it knows everything about it, so it should get the chance first. Then its immediate parent also knows about the context, but a little bit less, and so on till the very top element that handles general concepts and runs the last one. Bubbling and capturing lay the foundation for "event delegation" -- an extremely powerful event handling pattern that we study in the next chapter. diff --git a/2-ui/2-events/02-bubbling-and-capturing/event-order-bubbling.svg b/2-ui/2-events/02-bubbling-and-capturing/event-order-bubbling.svg index b62d99089..3339ecb97 100644 --- a/2-ui/2-events/02-bubbling-and-capturing/event-order-bubbling.svg +++ b/2-ui/2-events/02-bubbling-and-capturing/event-order-bubbling.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="353" height="216" viewBox="0 0 353 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="event-order-bubbling.svg"><path id="Rectangle-210" fill="#DBAF88" d="M159.488 140L174 186H60l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M173.634 81l16.09 51H43.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M39.986 29h153.028l22.71 72H17.276l22.71-72z"/><path id="Fill-46" fill="#166388" d="M121.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM117 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L121.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="1" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="210" y="105">1</tspan></text><text id="2" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="185" y="136">2</tspan></text><text id="3" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="157" y="181">3</tspan></text><text id="Most-deeply-nested-e" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="233.48" y="175">Most deeply</tspan> <tspan x="221.306" y="194">nested element</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M179.5 177.5h30"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="353" height="216" viewBox="0 0 353 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="event-order-bubbling.svg"><path id="Rectangle-210" fill="#DBAF88" d="M159.488 140L174 186H60l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M173.634 81l16.09 51H43.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M39.986 29h153.028l22.71 72H17.276l22.71-72z"/><path id="Fill-46" fill="#166388" d="M121.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM117 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L121.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="1" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="210" y="105">1</tspan></text><text id="2" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="185" y="136">2</tspan></text><text id="3" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="157" y="181">3</tspan></text><text id="Most-deeply-nested-e" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="233.48" y="175">Most deeply</tspan> <tspan x="221.306" y="194">nested element</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M179.5 177.5h30"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="353" height="216" viewBox="0 0 353 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="event-order-bubbling.svg"><path id="Rectangle-210" fill="#DBAF88" d="M159.488 140L174 186H60l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M173.634 81l16.09 51H43.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M193.014 29l22.71 72H17.276l22.71-72h153.028z"/><path id="Fill-46" fill="#166388" d="M121.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM117 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L121.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="1" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="210" y="105">1</tspan></text><text id="2" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="185" y="136">2</tspan></text><text id="3" fill="#C06334" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="157" y="181">3</tspan></text><text id="Most-deeply-nested-e" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="233.48" y="175">Most deeply</tspan> <tspan x="221.306" y="194">nested element</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M179.5 177.5h30"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/2-events/03-event-delegation/4-behavior-tooltip/task.md b/2-ui/2-events/03-event-delegation/4-behavior-tooltip/task.md index fc2462289..3001b9915 100644 --- a/2-ui/2-events/03-event-delegation/4-behavior-tooltip/task.md +++ b/2-ui/2-events/03-event-delegation/4-behavior-tooltip/task.md @@ -18,12 +18,14 @@ Should work like this: [iframe src="solution" height=200 border=1] -In this task we assume that all elements with `data-tooltip` have only text inside. No nested tags. +In this task we assume that all elements with `data-tooltip` have only text inside. No nested tags (yet). Details: +- The distance between the element and the tooltip should be `5px`. +- The tooltip should be centered relative to the element, if possible. - The tooltip should not cross window edges. Normally it should be above the element, but if the element is at the page top and there's no space for the tooltip, then below it. -- The tooltip is given in the `data-tooltip` attribute. It can be arbitrary HTML. +- The tooltip content is given in the `data-tooltip` attribute. It can be arbitrary HTML. You'll need two events here: - `mouseover` triggers when a pointer comes over an element. @@ -33,4 +35,4 @@ Please use event delegation: set up two handlers on `document` to track all "ove After the behavior is implemented, even people unfamiliar with JavaScript can add annotated elements. -P.S. To keep things natural and simple: only one tooltip may show up at a time. +P.S. Only one tooltip may show up at a time. diff --git a/2-ui/2-events/03-event-delegation/article.md b/2-ui/2-events/03-event-delegation/article.md index 8333e6bd1..881183740 100644 --- a/2-ui/2-events/03-event-delegation/article.md +++ b/2-ui/2-events/03-event-delegation/article.md @@ -1,11 +1,11 @@ # Event delegation -Capturing and bubbling allow us to implement one of most powerful event handling patterns called *event delegation*. +Capturing and bubbling allow us to implement one of the most powerful event handling patterns called *event delegation*. The idea is that if we have a lot of elements handled in a similar way, then instead of assigning a handler to each of them -- we put a single handler on their common ancestor. -In the handler we get `event.target`, see where the event actually happened and handle it. +In the handler we get `event.target` to see where the event actually happened and handle it. Let's see an example -- the [Ba-Gua diagram](http://en.wikipedia.org/wiki/Ba_gua) reflecting the ancient Chinese philosophy. @@ -21,9 +21,9 @@ The HTML is like this: <th colspan="3"><em>Bagua</em> Chart: Direction, Element, Color, Meaning</th> </tr> <tr> - <td>...<strong>Northwest</strong>...</td> - <td>...</td> - <td>...</td> + <td class="nw"><strong>Northwest</strong><br>Metal<br>Silver<br>Elders</td> + <td class="n">...</td> + <td class="ne">...</td> </tr> <tr>...2 more lines of this kind...</tr> <tr>...2 more lines of this kind...</tr> @@ -101,17 +101,17 @@ table.onclick = function(event) { Explanations: 1. The method `elem.closest(selector)` returns the nearest ancestor that matches the selector. In our case we look for `<td>` on the way up from the source element. -2. If `event.target` is not inside any `<td>`, then the call returns `null`, and we don't have to do anything. -3. In case of nested tables, `event.target` may be a `<td>` lying outside of the current table. So we check if that's actually *our table's* `<td>`. +2. If `event.target` is not inside any `<td>`, then the call returns immediately, as there's nothing to do. +3. In case of nested tables, `event.target` may be a `<td>`, but lying outside of the current table. So we check if that's actually *our table's* `<td>`. 4. And, if it's so, then highlight it. -## Delegation example: actions in markup +As the result, we have a fast, efficient highlighting code, that doesn't care about the total number of `<td>` in the table. -The event delegation may be used to optimize event handling. We use a single handler for similar actions on many elements. Like we did it for highlighting `<td>`. +## Delegation example: actions in markup -But we can also use a single handler as an entry point for many different things. +There are other uses for event delegation. -For instance, we want to make a menu with buttons "Save", "Load", "Search" and so on. And there's an object with methods `save`, `load`, `search`.... +Let's say, we want to make a menu with buttons "Save", "Load", "Search" and so on. And there's an object with methods `save`, `load`, `search`... How to match them? The first idea may be to assign a separate handler to each button. But there's a more elegant solution. We can add a handler for the whole menu and `data-action` attributes for buttons that has the method to call: @@ -121,7 +121,7 @@ The first idea may be to assign a separate handler to each button. But there's a The handler reads the attribute and executes the method. Take a look at the working example: -```html autorun height=60 run +```html autorun height=60 run untrusted <div id="menu"> <button data-action="save">Save</button> <button data-action="load">Load</button> @@ -161,9 +161,9 @@ The handler reads the attribute and executes the method. Take a look at the work </script> ``` -Please note that `this.onClick` is bound to `this` in `(*)`. That's important, because otherwise `this` inside it would reference the DOM element (`elem`), not the menu object, and `this[action]` would not be what we need. +Please note that `this.onClick` is bound to `this` in `(*)`. That's important, because otherwise `this` inside it would reference the DOM element (`elem`), not the `Menu` object, and `this[action]` would not be what we need. -So, what the delegation gives us here? +So, what advantages does delegation give us here? ```compare + We don't need to write the code to assign a handler to each button. Just make a method and put it in the markup. @@ -177,12 +177,12 @@ We could also use classes `.action-save`, `.action-load`, but an attribute `data We can also use event delegation to add "behaviors" to elements *declaratively*, with special attributes and classes. The pattern has two parts: -1. We add a special attribute to an element. +1. We add a custom attribute to an element that describes its behavior. 2. A document-wide handler tracks events, and if an event happens on an attributed element -- performs the action. -### Counter +### Behavior: Counter -For instance, here the attribute `data-counter` adds a behavior: "increase on click" to buttons: +For instance, here the attribute `data-counter` adds a behavior: "increase value on click" to buttons: ```html run autorun height=60 Counter: <input type="button" value="1" data-counter> @@ -204,14 +204,14 @@ If we click a button -- its value is increased. Not buttons, but the general app There can be as many attributes with `data-counter` as we want. We can add new ones to HTML at any moment. Using the event delegation we "extended" HTML, added an attribute that describes a new behavior. ```warn header="For document-level handlers -- always `addEventListener`" -When we assign an event handler to the `document` object, we should always use `addEventListener`, not `document.onclick`, because the latter will cause conflicts: new handlers overwrite old ones. +When we assign an event handler to the `document` object, we should always use `addEventListener`, not `document.on<event>`, because the latter will cause conflicts: new handlers overwrite old ones. For real projects it's normal that there are many handlers on `document` set by different parts of the code. ``` -### Toggler +### Behavior: Toggler -One more example. A click on an element with the attribute `data-toggle-id` will show/hide the element with the given `id`: +One more example of behavior. A click on an element with the attribute `data-toggle-id` will show/hide the element with the given `id`: ```html autorun run height=60 <button *!*data-toggle-id="subscribe-mail"*/!*> @@ -242,13 +242,13 @@ That may become really convenient -- no need to write JavaScript for every such We can combine multiple behaviors on a single element as well. -The "behavior" pattern can be an alternative of mini-fragments of JavaScript. +The "behavior" pattern can be an alternative to mini-fragments of JavaScript. ## Summary Event delegation is really cool! It's one of the most helpful patterns for DOM events. -It's often used to add same handling for many similar elements, but not only for that. +It's often used to add the same handling for many similar elements, but not only for that. The algorithm: @@ -261,12 +261,12 @@ Benefits: ```compare + Simplifies initialization and saves memory: no need to add many handlers. + Less code: when adding or removing elements, no need to add/remove handlers. -+ DOM modifications: we can mass add/remove elements with `innerHTML` and alike. ++ DOM modifications: we can mass add/remove elements with `innerHTML` and the like. ``` The delegation has its limitations of course: ```compare - First, the event must be bubbling. Some events do not bubble. Also, low-level handlers should not use `event.stopPropagation()`. -- Second, the delegation may add CPU load, because the container-level handler reacts on events in any place of the container, no matter if they interest us or not. But usually the load is negligible, so we don't take it into account. +- Second, the delegation may add CPU load, because the container-level handler reacts on events in any place of the container, no matter whether they interest us or not. But usually the load is negligible, so we don't take it into account. ``` diff --git a/2-ui/2-events/03-event-delegation/bagua-bubble.svg b/2-ui/2-events/03-event-delegation/bagua-bubble.svg index c4cd1ee1e..83086ed3f 100644 --- a/2-ui/2-events/03-event-delegation/bagua-bubble.svg +++ b/2-ui/2-events/03-event-delegation/bagua-bubble.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="369" height="216" viewBox="0 0 369 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bagua-bubble.svg"><path id="Rectangle-210" fill="#DBAF88" d="M202.488 140L217 186H103l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M216.634 81l16.09 51H86.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M82.986 29h153.028l22.71 72H60.276l22.71-72z"/><path id="Fill-46" fill="#166388" d="M164.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM160 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L164.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="<table>" fill="#D35155" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="9" y="30"><table></tspan></text><text id="<td>" fill="#478964" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="36" y="136"><td></tspan></text><text id="<strong>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="35" y="181"><strong></tspan></text><text id="event.target" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="271.537" y="181">event.target</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M229.5 177.5h30"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="369" height="216" viewBox="0 0 369 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bagua-bubble.svg"><path id="Rectangle-210" fill="#DBAF88" d="M202.488 140L217 186H103l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M216.634 81l16.09 51H86.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M82.986 29h153.028l22.71 72H60.276l22.71-72z"/><path id="Fill-46" fill="#166388" d="M164.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM160 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L164.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="<table>" fill="#D35155" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="9" y="30"><table></tspan></text><text id="<td>" fill="#478964" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="36" y="136"><td></tspan></text><text id="<strong>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="35" y="181"><strong></tspan></text><text id="event.target" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="271.537" y="181">event.target</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M229.5 177.5h30"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="369" height="216" viewBox="0 0 369 216"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bagua-bubble.svg"><path id="Rectangle-210" fill="#DBAF88" d="M202.488 140L217 186H103l14.512-46z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M216.634 81l16.09 51H86.276l16.09-51h114.267z"/><path id="Rectangle-208" stroke="#EFA39F" stroke-width="18" d="M236.014 29l22.71 72H60.276l22.71-72h153.028z"/><path id="Fill-46" fill="#166388" d="M164.5 141v13.816a4.5 4.5 0 11-9 0V141h9zm0-31v13h-9v-13h9zM160 20.53a4.471 4.471 0 013.362 1.3l17.64 17.64a4.5 4.5 0 01-6.364 6.364L164.5 35.698V92h-9V35.7l-10.138 10.136a4.5 4.5 0 11-6.363-6.364l17.639-17.64a4.477 4.477 0 013.363-1.3z"/><text id="<table>" fill="#D35155" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="9" y="30"><table></tspan></text><text id="<td>" fill="#478964" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="36" y="136"><td></tspan></text><text id="<strong>" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="35" y="181"><strong></tspan></text><text id="event.target" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="271.537" y="181">event.target</tspan></text><path id="Line-30" stroke="#C06334" stroke-dasharray="3,6" stroke-linecap="square" stroke-width="2" d="M229.5 177.5h30"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/solution.md b/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/solution.md index 2862e6006..4d175ca01 100644 --- a/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/solution.md +++ b/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/solution.md @@ -20,7 +20,7 @@ The fix is simple: } </script> -<a href="http://w3.org" onclick="*!*return handler()*/!*">w3.org</a> +<a href="https://w3.org" onclick="*!*return handler()*/!*">w3.org</a> ``` Also we can use `event.preventDefault()`, like this: @@ -35,5 +35,5 @@ Also we can use `event.preventDefault()`, like this: */!* </script> -<a href="http://w3.org" onclick="*!*handler(event)*/!*">w3.org</a> +<a href="https://w3.org" onclick="*!*handler(event)*/!*">w3.org</a> ``` diff --git a/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/task.md b/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/task.md index 8fa4fb29a..9a5bb98ac 100644 --- a/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/task.md +++ b/2-ui/2-events/04-default-browser-action/1-why-return-false-fails/task.md @@ -14,7 +14,7 @@ Why in the code below `return false` doesn't work at all? } </script> -<a href="http://w3.org" onclick="handler()">the browser will go to w3.org</a> +<a href="https://w3.org" onclick="handler()">the browser will go to w3.org</a> ``` The browser follows the URL on click, but we don't want it. diff --git a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/solution.view/index.html b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/solution.view/index.html index 90aecd9dc..51ac0838b 100644 --- a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/solution.view/index.html +++ b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/solution.view/index.html @@ -16,7 +16,7 @@ <fieldset id="contents"> <legend>#contents</legend> <p> - How about to read <a href="http://wikipedia.org">Wikipedia</a> or visit <a href="http://w3.org"><i>W3.org</i></a> and learn about modern standards? + How about to read <a href="https://wikipedia.org">Wikipedia</a> or visit <a href="https://w3.org"><i>W3.org</i></a> and learn about modern standards? </p> </fieldset> diff --git a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/source.view/index.html b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/source.view/index.html index b88834c96..f0c934391 100644 --- a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/source.view/index.html +++ b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/source.view/index.html @@ -16,7 +16,7 @@ <fieldset id="contents"> <legend>#contents</legend> <p> - How about to read <a href="http://wikipedia.org">Wikipedia</a> or visit <a href="http://w3.org"><i>W3.org</i></a> and learn about modern standards? + How about to read <a href="https://wikipedia.org">Wikipedia</a> or visit <a href="https://w3.org"><i>W3.org</i></a> and learn about modern standards? </p> </fieldset> diff --git a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/task.md b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/task.md index 6439376bf..6ca456c2c 100644 --- a/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/task.md +++ b/2-ui/2-events/04-default-browser-action/2-catch-link-navigation/task.md @@ -12,5 +12,5 @@ Like this: Details: -- HTML inside the element may be loaded or regenerated dynamically at any time, so we can't find all links and put handlers on them. Use the event delegation. +- HTML inside the element may be loaded or regenerated dynamically at any time, so we can't find all links and put handlers on them. Use event delegation. - The content may have nested tags. Inside links too, like `<a href=".."><i>...</i></a>`. diff --git a/2-ui/2-events/04-default-browser-action/3-image-gallery/solution.view/gallery.css b/2-ui/2-events/04-default-browser-action/3-image-gallery/solution.view/gallery.css index 4522006ae..8d6472ee6 100644 --- a/2-ui/2-events/04-default-browser-action/3-image-gallery/solution.view/gallery.css +++ b/2-ui/2-events/04-default-browser-action/3-image-gallery/solution.view/gallery.css @@ -4,16 +4,6 @@ body { font: 75%/120% sans-serif; } -h2 { - font: bold 190%/100% sans-serif; - margin: 0 0 .2em; -} - -h2 em { - font: normal 80%/100% sans-serif; - color: #999999; -} - #largeImg { border: solid 1px #ccc; width: 550px; diff --git a/2-ui/2-events/04-default-browser-action/3-image-gallery/source.view/gallery.css b/2-ui/2-events/04-default-browser-action/3-image-gallery/source.view/gallery.css index b6e523014..8d6472ee6 100644 --- a/2-ui/2-events/04-default-browser-action/3-image-gallery/source.view/gallery.css +++ b/2-ui/2-events/04-default-browser-action/3-image-gallery/source.view/gallery.css @@ -4,16 +4,6 @@ body { font: 75%/120% sans-serif; } -h2 { - font: bold 190%/100% sans-serif; - margin: 0 0 .2em; -} - -h2 em { - font: normal 80%/100% sans-serif; - color: #999999; -} - #largeImg { border: solid 1px #ccc; width: 550px; @@ -32,4 +22,13 @@ h2 em { #thumbs a:hover { border-color: #FF9900; +} + +#thumbs li { + list-style: none; +} + +#thumbs { + margin: 0; + padding: 0; } \ No newline at end of file diff --git a/2-ui/2-events/04-default-browser-action/article.md b/2-ui/2-events/04-default-browser-action/article.md index 44355de0d..cd815654f 100644 --- a/2-ui/2-events/04-default-browser-action/article.md +++ b/2-ui/2-events/04-default-browser-action/article.md @@ -1,23 +1,23 @@ # Browser default actions -Many events automatically lead to browser actions. +Many events automatically lead to certain actions performed by the browser. For instance: -- A click on a link -- initiates going to its URL. -- A click on submit button inside a form -- initiates its submission to the server. -- Pressing a mouse button over a text and moving it -- selects the text. +- A click on a link - initiates navigation to its URL. +- A click on a form submit button - initiates its submission to the server. +- Pressing a mouse button over a text and moving it - selects the text. -If we handle an event in JavaScript, often we don't want browser actions. Fortunately, it can be prevented. +If we handle an event in JavaScript, we may not want the corresponding browser action to happen, and want to implement another behavior instead. ## Preventing browser actions There are two ways to tell the browser we don't want it to act: - The main way is to use the `event` object. There's a method `event.preventDefault()`. -- If the handler is assigned using `on<event>` (not by `addEventListener`), then we can just return `false` from it. +- If the handler is assigned using `on<event>` (not by `addEventListener`), then returning `false` also works the same. -In the example below a click to links doesn't lead to URL change: +In this HTML, a click on a link doesn't lead to navigation; the browser doesn't do anything: ```html autorun height=60 no-beautify <a href="/" onclick="return false">Click here</a> @@ -25,12 +25,14 @@ or <a href="/" onclick="event.preventDefault()">here</a> ``` -```warn header="Not necessary to return `true`" +In the next example we'll use this technique to create a JavaScript-powered menu. + +```warn header="Returning `false` from a handler is an exception" The value returned by an event handler is usually ignored. -The only exception -- is `return false` from a handler assigned using `on<event>`. +The only exception is `return false` from a handler assigned using `on<event>`. -In all other cases, the return is not needed and it's not processed anyhow. +In all other cases, `return` value is ignored. In particular, there's no sense in returning `true`. ``` ### Example: the menu @@ -49,7 +51,7 @@ Here's how it looks with some CSS: [iframe height=70 src="menu" link edit] -Menu items are links `<a>`, not buttons. There are several benefits, for instance: +Menu items are implemented as HTML-links `<a>`, not buttons `<button>`. There are several reasons to do so, for instance: - Many people like to use "right click" -- "open in a new window". If we use `<button>` or `<span>`, that doesn't work. - Search engines follow `<a href="...">` links while indexing. @@ -71,20 +73,16 @@ menu.onclick = function(event) { }; ``` -If we omit `return false`, then after our code executes the browser will do its "default action" -- following to the URL in `href`. - -By the way, using event delegation here makes our menu flexible. We can add nested lists and style them using CSS to "slide down". - +If we omit `return false`, then after our code executes the browser will do its "default action" -- navigating to the URL in `href`. And we don't need that here, as we're handling the click by ourselves. -## Prevent further events +By the way, using event delegation here makes our menu very flexible. We can add nested lists and style them using CSS to "slide down". +````smart header="Follow-up events" Certain events flow one into another. If we prevent the first event, there will be no second. For instance, `mousedown` on an `<input>` field leads to focusing in it, and the `focus` event. If we prevent the `mousedown` event, there's no focus. -Try to click on the first `<input>` below -- the `focus` event happens. That's normal. - -But if you click the second one, there's no focus. +Try to click on the first `<input>` below -- the `focus` event happens. But if you click the second one, there's no focus. ```html run autorun <input value="Focus works" onfocus="this.value=''"> @@ -92,12 +90,13 @@ But if you click the second one, there's no focus. ``` That's because the browser action is canceled on `mousedown`. The focusing is still possible if we use another way to enter the input. For instance, the `key:Tab` key to switch from the 1st input into the 2nd. But not with the mouse click any more. +```` ## The "passive" handler option The optional `passive: true` option of `addEventListener` signals the browser that the handler is not going to call `preventDefault()`. -Why that may be needed? +Why might that be needed? There are some events like `touchmove` on mobile devices (when the user moves their finger across the screen), that cause scrolling by default, but that scrolling can be prevented using `preventDefault()` in the handler. @@ -114,23 +113,25 @@ The property `event.defaultPrevented` is `true` if the default action was preven There's an interesting use case for it. -You remember in the chapter <info:bubbling-and-capturing> we talked about `event.stopPropagation()` and why stopping bubbling is bad? +You remember in the chapter <info:bubbling-and-capturing> we talked about `event.stopPropagation()` and why stopping bubbling is bad? -Sometimes we can use `event.defaultPrevented` instead. +Sometimes we can use `event.defaultPrevented` instead, to signal other event handlers that the event was handled. -Let's see a practical example where stopping the bubbling looks necessary, but actually we can do well without it. +Let's see a practical example. By default the browser on `contextmenu` event (right mouse click) shows a context menu with standard options. We can prevent it and show our own, like this: ```html autorun height=50 no-beautify run -<button>Right-click for browser context menu</button> +<button>Right-click shows browser context menu</button> <button *!*oncontextmenu="alert('Draw our menu'); return false"*/!*> - Right-click for our context menu + Right-click shows our context menu </button> ``` -Now let's say we want to implement our own document-wide context menu, with our options. And inside the document we may have other elements with their own context menus: +Now, in addition to that context menu we'd like to implement document-wide context menu. + +Upon right click, the closest context menu should show up. ```html autorun height=80 no-beautify run <p>Right-click here for the document context menu</p> @@ -151,7 +152,7 @@ Now let's say we want to implement our own document-wide context menu, with our The problem is that when we click on `elem`, we get two menus: the button-level and (the event bubbles up) the document-level menu. -How to fix it? One of solutions is to think like: "We fully handle the event in the button handler, let's stop it" and use `event.stopPropagation()`: +How to fix it? One of solutions is to think like: "When we handle right-click in the button handler, let's stop its bubbling" and use `event.stopPropagation()`: ```html autorun height=80 no-beautify run <p>Right-click for the document menu</p> @@ -179,7 +180,7 @@ An alternative solution would be to check in the `document` handler if the defau ```html autorun height=80 no-beautify run -<p>Right-click for the document menu (fixed with event.defaultPrevented)</p> +<p>Right-click for the document menu (added a check for event.defaultPrevented)</p> <button id="elem">Right-click for the button menu</button> <script> @@ -206,7 +207,7 @@ As we can clearly see, `event.stopPropagation()` and `event.preventDefault()` (a ``` ```smart header="Nested context menus architecture" -There are also alternative ways to implement nested context menus. One of them is to have a special global object with a method that handles `document.oncontextmenu`, and also methods that allow to store various "lower-level" handlers in it. +There are also alternative ways to implement nested context menus. One of them is to have a single global object with a handler for `document.oncontextmenu`, and also methods that allow us to store other handlers in it. The object will catch any right-click, look through stored handlers and run the appropriate one. @@ -220,7 +221,6 @@ There are many default browser actions: - `mousedown` -- starts the selection (move the mouse to select). - `click` on `<input type="checkbox">` -- checks/unchecks the `input`. - `submit` -- clicking an `<input type="submit">` or hitting `key:Enter` inside a form field causes this event to happen, and the browser submits the form after it. -- `wheel` -- rolling a mouse wheel event has scrolling as the default action. - `keydown` -- pressing a key may lead to adding a character into a field, or other actions. - `contextmenu` -- the event happens on a right-click, the action is to show the browser context menu. - ...there are more... @@ -236,9 +236,9 @@ If the default action was prevented, the value of `event.defaultPrevented` becom ```warn header="Stay semantic, don't abuse" Technically, by preventing default actions and adding JavaScript we can customize the behavior of any elements. For instance, we can make a link `<a>` work like a button, and a button `<button>` behave as a link (redirect to another URL or so). -But we should generally keep the semantic meaning of HTML elements. For instance, `<a>` should preform navigation, not a button. +But we should generally keep the semantic meaning of HTML elements. For instance, `<a>` should perform navigation, not a button. Besides being "just a good thing", that makes your HTML better in terms of accessibility. -Also if we consider the example with `<a>`, then please note: a browser allows to open such links in a new window (by right-clicking them and other means). And people like that. But if we make a button behave as a link using JavaScript and even look like a link using CSS, then `<a>`-specific browser features still won't work for it. +Also if we consider the example with `<a>`, then please note: a browser allows us to open such links in a new window (by right-clicking them and other means). And people like that. But if we make a button behave as a link using JavaScript and even look like a link using CSS, then `<a>`-specific browser features still won't work for it. ``` diff --git a/2-ui/2-events/05-dispatch-events/article.md b/2-ui/2-events/05-dispatch-events/article.md index 2b13d2360..047413fd3 100644 --- a/2-ui/2-events/05-dispatch-events/article.md +++ b/2-ui/2-events/05-dispatch-events/article.md @@ -2,23 +2,23 @@ We can not only assign handlers, but also generate events from JavaScript. -Custom events can be used to create "graphical components". For instance, a root element of the menu may trigger events telling what happens with the menu: `open` (menu open), `select` (an item is selected) and so on. +Custom events can be used to create "graphical components". For instance, a root element of our own JS-based menu may trigger events telling what happens with the menu: `open` (menu open), `select` (an item is selected) and so on. Another code may listen for the events and observe what's happening with the menu. -Also we can generate built-in events like `click`, `mousedown` etc, that may be good for testing. +We can generate not only completely new events, that we invent for our own purposes, but also built-in ones, such as `click`, `mousedown` etc. That may be helpful for automated testing. ## Event constructor -Events form a hierarchy, just like DOM element classes. The root is the built-in [Event](http://www.w3.org/TR/dom/#event) class. +Built-in event classes form a hierarchy, similar to DOM element classes. The root is the built-in [Event](https://dom.spec.whatwg.org/#events) class. We can create `Event` objects like this: ```js -let event = new Event(event type[, options]); +let event = new Event(type[, options]); ``` Arguments: -- *event type* -- may be any string, like `"click"` or our own like `"hey-ho!"`. +- *type* -- event type, a string like `"click"` or our own like `"my-event"`. - *options* -- the object with two optional properties: - `bubbles: true/false` -- if `true`, then the event bubbles. - `cancelable: true/false` -- if `true`, then the "default action" may be prevented. Later we'll see what it means for custom events. @@ -27,9 +27,9 @@ Arguments: ## dispatchEvent -After an event object is created, we should "run" it on an element using the call `elem.dispatchEvent(event)`. +After an event object is created, we should "run" it on an element using the call `elem.dispatchEvent(event)`. -Then handlers react on it as if it were a regular built-in event. If the event was created with the `bubbles` flag, then it bubbles. +Then handlers react on it as if it were a regular browser event. If the event was created with the `bubbles` flag, then it bubbles. In the example below the `click` event is initiated in JavaScript. The handler works same way as if the button was clicked: @@ -66,9 +66,13 @@ All we need is to set `bubbles` to `true`: // ...dispatch on elem! let event = new Event("hello", {bubbles: true}); // (2) elem.dispatchEvent(event); + + // the handler on document will activate and display the message. + </script> ``` + Notes: 1. We should use `addEventListener` for our custom events, because `on<event>` only exists for built-in events, `document.onhello` doesn't work. @@ -125,11 +129,11 @@ alert(event.clientX); // undefined, the unknown property is ignored! Technically, we can work around that by assigning directly `event.clientX=100` after creation. So that's a matter of convenience and following the rules. Browser-generated events always have the right type. -The full list of properties for different UI events is in the specification, for instance [MouseEvent](https://www.w3.org/TR/uievents/#mouseevent). +The full list of properties for different UI events is in the specification, for instance, [MouseEvent](https://www.w3.org/TR/uievents/#mouseevent). ## Custom events -For our own, custom events like `"hello"` we should use `new CustomEvent`. Technically [CustomEvent](https://dom.spec.whatwg.org/#customevent) is the same as `Event`, with one exception. +For our own, completely new events types like `"hello"` we should use `new CustomEvent`. Technically [CustomEvent](https://dom.spec.whatwg.org/#customevent) is the same as `Event`, with one exception. In the second argument (object) we can add an additional property `detail` for any custom information that we want to pass with the event. @@ -154,25 +158,25 @@ For instance: The `detail` property can have any data. Technically we could live without, because we can assign any properties into a regular `new Event` object after its creation. But `CustomEvent` provides the special `detail` field for it to evade conflicts with other event properties. -The event class tells something about "what kind of event" it is, and if the event is custom, then we should use `CustomEvent` just to be clear about what it is. +Besides, the event class describes "what kind of event" it is, and if the event is custom, then we should use `CustomEvent` just to be clear about what it is. ## event.preventDefault() -We can call `event.preventDefault()` on a script-generated event if `cancelable:true` flag is specified. +Many browser events have a "default action", such as navigating to a link, starting a selection, and so on. -Of course, if the event has a non-standard name, then it's not known to the browser, and there's no "default browser action" for it. +For new, custom events, there are definitely no default browser actions, but a code that dispatches such event may have its own plans what to do after triggering the event. -But the event-generating code may plan some actions after `dispatchEvent`. +By calling `event.preventDefault()`, an event handler may send a signal that those actions should be canceled. -The call of `event.preventDefault()` is a way for the handler to send a signal that those actions shouldn't be performed. +In that case the call to `elem.dispatchEvent(event)` returns `false`. And the code that dispatched it knows that it shouldn't continue. -In that case the call to `elem.dispatchEvent(event)` returns `false`. And the event-generating code knows that the processing shouldn't continue. +Let's see a practical example - a hiding rabbit (could be a closing menu or something else). -For instance, in the example below there's a `hide()` function. It generates the `"hide"` event on the element `#rabbit`, notifying all interested parties that the rabbit is going to hide. +Below you can see a `#rabbit` and `hide()` function that dispatches `"hide"` event on it, to let all interested parties know that the rabbit is going to hide. -A handler set by `rabbit.addEventListener('hide',...)` will learn about that and, if it wants, can prevent that action by calling `event.preventDefault()`. Then the rabbit won't hide: +Any handler can listen for that event with `rabbit.addEventListener('hide',...)` and, if needed, cancel the action using `event.preventDefault()`. Then the rabbit won't disappear: -```html run refresh +```html run refresh autorun <pre id="rabbit"> |\ /| \|_|/ @@ -180,15 +184,15 @@ A handler set by `rabbit.addEventListener('hide',...)` will learn about that and =\_Y_/= {>o<} </pre> +<button onclick="hide()">Hide()</button> <script> - // hide() will be called automatically in 2 seconds function hide() { let event = new CustomEvent("hide", { cancelable: true // without that flag preventDefault doesn't work }); if (!rabbit.dispatchEvent(event)) { - alert('the action was prevented by a handler'); + alert('The action was prevented by a handler'); } else { rabbit.hidden = true; } @@ -199,33 +203,29 @@ A handler set by `rabbit.addEventListener('hide',...)` will learn about that and event.preventDefault(); } }); - - // hide in 2 seconds - setTimeout(hide, 2000); - </script> ``` +Please note: the event must have the flag `cancelable: true`, otherwise the call `event.preventDefault()` is ignored. ## Events-in-events are synchronous -Usually events are processed asynchronously. That is: if the browser is processing `onclick` and in the process a new event occurs, then it awaits till `onclick` processing is finished. +Usually events are processed in a queue. That is: if the browser is processing `onclick` and a new event occurs, e.g. mouse moved, then its handling is queued up, corresponding `mousemove` handlers will be called after `onclick` processing is finished. -The exception is when one event is initiated from within another one. +The notable exception is when one event is initiated from within another one, e.g. using `dispatchEvent`. Such events are processed immediately: the new event handlers are called, and then the current event handling is resumed. -Then the control jumps to the nested event handler, and after it goes back. +For instance, in the code below the `menu-open` event is triggered during the `onclick`. -For instance, here the nested `menu-open` event is processed synchronously, during the `onclick`: +It's processed immediately, without waiting for `onclick` handler to end: -```html run + +```html run autorun <button id="menu">Menu (click me)</button> <script> - // 1 -> nested -> 2 menu.onclick = function() { alert(1); - // alert("nested") menu.dispatchEvent(new CustomEvent("menu-open", { bubbles: true })); @@ -233,51 +233,58 @@ For instance, here the nested `menu-open` event is processed synchronously, duri alert(2); }; - document.addEventListener('menu-open', () => alert('nested')) + // triggers between 1 and 2 + document.addEventListener('menu-open', () => alert('nested')); </script> -``` +``` + +The output order is: 1 -> nested -> 2. + +Please note that the nested event `menu-open` is caught on the `document`. The propagation and handling of the nested event is finished before the processing gets back to the outer code (`onclick`). -Please note that the nested event `menu-open` bubbles up and is handled on the `document`. The propagation of the nested event is fully finished before the processing gets back to the outer code (`onclick`). +That's not only about `dispatchEvent`, there are other cases. If an event handler calls methods that trigger other events -- they are processed synchronously too, in a nested fashion. -That's not only about `dispatchEvent`, there are other cases. JavaScript in an event handler can call methods that lead to other events -- they are too processed synchronously. +Let's say we don't like it. We'd want `onclick` to be fully processed first, independently from `menu-open` or any other nested events. -If we don't like it, we can either put the `dispatchEvent` (or other event-triggering call) at the end of `onclick` or, if inconvenient, wrap it in `setTimeout(...,0)`: +Then we can either put the `dispatchEvent` (or another event-triggering call) at the end of `onclick` or, maybe better, wrap it in the zero-delay `setTimeout`: ```html run <button id="menu">Menu (click me)</button> <script> - // 1 -> 2 -> nested menu.onclick = function() { alert(1); - // alert(2) setTimeout(() => menu.dispatchEvent(new CustomEvent("menu-open", { bubbles: true - })), 0); + }))); alert(2); }; - document.addEventListener('menu-open', () => alert('nested')) + document.addEventListener('menu-open', () => alert('nested')); </script> -``` +``` + +Now `dispatchEvent` runs asynchronously after the current code execution is finished, including `menu.onclick`, so event handlers are totally separate. + +The output order becomes: 1 -> 2 -> nested. ## Summary -To generate an event, we first need to create an event object. +To generate an event from code, we first need to create an event object. The generic `Event(name, options)` constructor accepts an arbitrary event name and the `options` object with two properties: - - `bubbles: true` if the event should bubble. - - `cancelable: true` if the `event.preventDefault()` should work. +- `bubbles: true` if the event should bubble. +- `cancelable: true` if the `event.preventDefault()` should work. Other constructors of native events like `MouseEvent`, `KeyboardEvent` and so on accept properties specific to that event type. For instance, `clientX` for mouse events. For custom events we should use `CustomEvent` constructor. It has an additional option named `detail`, we should assign the event-specific data to it. Then all handlers can access it as `event.detail`. -Despite the technical possibility to generate browser events like `click` or `keydown`, we should use with the great care. +Despite the technical possibility of generating browser events like `click` or `keydown`, we should use them with great care. -We shouldn't generate browser events as it's a hacky way to run handlers. That's a bad architecture most of the time. +We shouldn't generate browser events as it's a hacky way to run handlers. That's bad architecture most of the time. Native events might be generated: diff --git a/2-ui/3-event-details/1-mouse-events-basics/01-selectable-list/task.md b/2-ui/3-event-details/1-mouse-events-basics/01-selectable-list/task.md index f358616ef..8d29134ff 100644 --- a/2-ui/3-event-details/1-mouse-events-basics/01-selectable-list/task.md +++ b/2-ui/3-event-details/1-mouse-events-basics/01-selectable-list/task.md @@ -14,4 +14,5 @@ The demo: [iframe border="1" src="solution" height=180] P.S. For this task we can assume that list items are text-only. No nested tags. + P.P.S. Prevent the native browser selection of the text on clicks. diff --git a/2-ui/3-event-details/1-mouse-events-basics/article.md b/2-ui/3-event-details/1-mouse-events-basics/article.md index dce1ad3de..9574b0c83 100644 --- a/2-ui/3-event-details/1-mouse-events-basics/article.md +++ b/2-ui/3-event-details/1-mouse-events-basics/article.md @@ -1,16 +1,13 @@ -# Mouse events basics -Mouse events come not only from "mouse manipulators", but are also emulated on touch devices, to make them compatible. +# Mouse events In this chapter we'll get into more details about mouse events and their properties. -## Mouse event types - -We can split mouse events into two categories: "simple" and "complex" +Please note: such events may come not only from "mouse devices", but are also from other devices, such as phones and tablets, where they are emulated for compatibility. -### Simple events +## Mouse event types -The most used simple events are: +We've already seen some of these events: `mousedown/mouseup` : Mouse button is clicked/released over an element. @@ -21,65 +18,79 @@ The most used simple events are: `mousemove` : Every mouse move over an element triggers that event. -...There are several other event types too, we'll cover them later. - -### Complex events - `click` : Triggers after `mousedown` and then `mouseup` over the same element if the left mouse button was used. -`contextmenu` -: Triggers after `mousedown` if the right mouse button was used. - `dblclick` -: Triggers after a double click over an element. +: Triggers after two clicks on the same element within a short timeframe. Rarely used nowadays. + +`contextmenu` +: Triggers when the right mouse button is pressed. There are other ways to open a context menu, e.g. using a special keyboard key, it triggers in that case also, so it's not exactly the mouse event. -Complex events are made of simple ones, so in theory we could live without them. But they exist, and that's good, because they are convenient. +...There are several other events too, we'll cover them later. -### Events order +## Events order -An action may trigger multiple events. +As you can see from the list above, a user action may trigger multiple events. -For instance, a click first triggers `mousedown`, when the button is pressed, then `mouseup` and `click` when it's released. +For instance, a left-button click first triggers `mousedown`, when the button is pressed, then `mouseup` and `click` when it's released. -In cases when a single action initiates multiple events, their order is fixed. That is, the handlers are called in the order `mousedown` -> `mouseup` -> `click`. Events are handled in the same sequence: `onmouseup` finishes before `onclick` runs. +In cases when a single action initiates multiple events, their order is fixed. That is, the handlers are called in the order `mousedown` -> `mouseup` -> `click`. ```online Click the button below and you'll see the events. Try double-click too. -On the teststand below all mouse events are logged, and if there are more than 1 second delay between them, then they are separated by a horizontal ruler. +On the teststand below, all mouse events are logged, and if there is more than a 1 second delay between them, they are separated by a horizontal rule. -Also we can see the `which` property that allows to detect the mouse button. +Also, we can see the `button` property that allows us to detect the mouse button; it's explained below. <input onmousedown="return logMouse(event)" onmouseup="return logMouse(event)" onclick="return logMouse(event)" oncontextmenu="return logMouse(event)" ondblclick="return logMouse(event)" value="Click me with the right or the left mouse button" type="button"> <input onclick="logClear('test')" value="Clear" type="button"> <form id="testform" name="testform"> <textarea style="font-size:12px;height:150px;width:360px;"></textarea></form> ``` -## Getting the button: which +## Mouse button + +Click-related events always have the `button` property, which allows to get the exact mouse button. + +We usually don't use it for `click` and `contextmenu` events, because the former happens only on left-click, and the latter -- only on right-click. + +On the other hand, `mousedown` and `mouseup` handlers may need `event.button`, because these events trigger on any button, so `button` allows to distinguish between "right-mousedown" and "left-mousedown". -Click-related events always have the `which` property, which allows to get the exact mouse button. +The possible values of `event.button` are: -It is not used for `click` and `contextmenu` events, because the former happens only on left-click, and the latter -- only on right-click. +| Button state | `event.button` | +|--------------|----------------| +| Left button (primary) | 0 | +| Middle button (auxiliary) | 1 | +| Right button (secondary) | 2 | +| X1 button (back) | 3 | +| X2 button (forward) | 4 | -But if we track `mousedown` and `mouseup`, then we need it, because these events trigger on any button, so `which` allows to distinguish between "right-mousedown" and "left-mousedown". +Most mouse devices only have the left and right buttons, so possible values are `0` or `2`. Touch devices also generate similar events when one taps on them. -There are the three possible values: +Also there's `event.buttons` property that has all currently pressed buttons as an integer, one bit per button. In practice this property is very rarely used, you can find details at [MDN](mdn:/api/MouseEvent/buttons) if you ever need it. -- `event.which == 1` -- the left button -- `event.which == 2` - the middle button -- `event.which == 3` - the right button +```warn header="The outdated `event.which`" +Old code may use `event.which` property that's an old non-standard way of getting a button, with possible values: -The middle button is somewhat exotic right now and is very rarely used. +- `event.which == 1` – left button, +- `event.which == 2` – middle button, +- `event.which == 3` – right button. + +As of now, `event.which` is deprecated, we shouldn't use it. +``` ## Modifiers: shift, alt, ctrl and meta All mouse events include the information about pressed modifier keys. -The properties are: +Event properties: + +- `shiftKey`: `key:Shift` +- `altKey`: `key:Alt` (or `key:Opt` for Mac) +- `ctrlKey`: `key:Ctrl` +- `metaKey`: `key:Cmd` for Mac -- `shiftKey` -- `altKey` -- `ctrlKey` -- `metaKey` (`key:Cmd` for Mac) +They are `true` if the corresponding key was pressed during the event. For instance, the button below only works on `key:Alt+Shift`+click: @@ -98,88 +109,65 @@ For instance, the button below only works on `key:Alt+Shift`+click: ``` ```warn header="Attention: on Mac it's usually `Cmd` instead of `Ctrl`" -On Windows and Linux there are modifier keys `key:Alt`, `key:Shift` and `key:Ctrl`. On Mac there's one more: `key:Cmd`, it corresponds to the property `metaKey`. +On Windows and Linux there are modifier keys `key:Alt`, `key:Shift` and `key:Ctrl`. On Mac there's one more: `key:Cmd`, corresponding to the property `metaKey`. -In most cases when Windows/Linux uses `key:Ctrl`, on Mac people use `key:Cmd`. So where a Windows user presses `key:Ctrl+Enter` or `key:Ctrl+A`, a Mac user would press `key:Cmd+Enter` or `key:Cmd+A`, and so on, most apps use `key:Cmd` instead of `key:Ctrl`. +In most applications, when Windows/Linux uses `key:Ctrl`, on Mac `key:Cmd` is used. -So if we want to support combinations like `key:Ctrl`+click, then for Mac it makes sense to use `key:Cmd`+click. That's more comfortable for Mac users. +That is: where a Windows user presses `key:Ctrl+Enter` or `key:Ctrl+A`, a Mac user would press `key:Cmd+Enter` or `key:Cmd+A`, and so on. -Even if we'd like to force Mac users to `key:Ctrl`+click -- that's kind of difficult. The problem is: a left-click with `key:Ctrl` is interpreted as a *right-click* on Mac, and it generates the `contextmenu` event, not `click` like Windows/Linux. +So if we want to support combinations like `key:Ctrl`+click, then for Mac it makes sense to use `key:Cmd`+click. That's more comfortable for Mac users. -So if we want users of all operational systems to feel comfortable, then together with `ctrlKey` we should use `metaKey`. +Even if we'd like to force Mac users to `key:Ctrl`+click -- that's kind of difficult. The problem is: a left-click with `key:Ctrl` is interpreted as a *right-click* on MacOS, and it generates the `contextmenu` event, not `click` like Windows/Linux. + +So if we want users of all operating systems to feel comfortable, then together with `ctrlKey` we should check `metaKey`. For JS-code it means that we should check `if (event.ctrlKey || event.metaKey)`. ``` ```warn header="There are also mobile devices" -Keyboard combinations are good as an addition to the workflow. So that if the visitor has a - keyboard -- it works. And if your device doesn't have it -- then there's another way to do the same. +Keyboard combinations are good as an addition to the workflow. So that if the visitor uses a keyboard -- they work. + +But if their device doesn't have it -- then there should be a way to live without modifier keys. ``` ## Coordinates: clientX/Y, pageX/Y -All mouse events have coordinates in two flavours: +All mouse events provide coordinates in two flavours: 1. Window-relative: `clientX` and `clientY`. 2. Document-relative: `pageX` and `pageY`. -For instance, if we have a window of the size 500x500, and the mouse is in the left-upper corner, then `clientX` and `clientY` are `0`. And if the mouse is in the center, then `clientX` and `clientY` are `250`, no matter what place in the document it is. They are similar to `position:fixed`. +We already covered the difference between them in the chapter <info:coordinates>. + +In short, document-relative coordinates `pageX/Y` are counted from the left-upper corner of the document, and do not change when the page is scrolled, while `clientX/Y` are counted from the current window left-upper corner. When the page is scrolled, they change. + +For instance, if we have a window of the size 500x500, and the mouse is in the left-upper corner, then `clientX` and `clientY` are `0`, no matter how the page is scrolled. + +And if the mouse is in the center, then `clientX` and `clientY` are `250`, no matter what place in the document it is. They are similar to `position:fixed` in that aspect. ````online -Move the mouse over the input field to see `clientX/clientY` (it's in the `iframe`, so coordinates are relative to that `iframe`): +Move the mouse over the input field to see `clientX/clientY` (the example is in the `iframe`, so coordinates are relative to that `iframe`): ```html autorun height=50 <input onmousemove="this.value=event.clientX+':'+event.clientY" value="Mouse over me"> ``` ```` -Document-relative coordinates are counted from the left-upper corner of the document, not the window. -Coordinates `pageX`, `pageY` are similar to `position:absolute` on the document level. - -You can read more about coordinates in the chapter <info:coordinates>. - -## No selection on mousedown - -Mouse clicks have a side-effect that may be disturbing. A double click selects the text. +## Preventing selection on mousedown -If we want to handle click events ourselves, then the "extra" selection doesn't look good. +Double mouse click has a side effect that may be disturbing in some interfaces: it selects text. -For instance, a double-click on the text below selects it in addition to our handler: +For instance, double-clicking on the text below selects it in addition to our handler: ```html autorun height=50 -<b ondblclick="alert('dblclick')">Double-click me</b> +<span ondblclick="alert('dblclick')">Double-click me</span> ``` -There's a CSS way to stop the selection: the `user-select` property from [CSS UI Draft](https://www.w3.org/TR/css-ui-4/). +If one presses the left mouse button and, without releasing it, moves the mouse, that also makes the selection, often unwanted. -Most browsers support it with prefixes: +There are multiple ways to prevent the selection, that you can read in the chapter <info:selection-range>. -```html autorun height=50 -<style> - b { - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - } -</style> - -Before... -<b ondblclick="alert('Test')"> - Unselectable -</b> -...After -``` - -Now if you double-click on "Unselectable", it doesn't get selected. Seems to work. - -...But there is a potential problem! The text became truly unselectable. Even if a user starts the selection from "Before" and ends with "After", the selection skips "Unselectable" part. Do we really want to make our text unselectable? - -Most of time, we don't. A user may have valid reasons to select the text, for copying or other needs. That may be inconvenient if we don't allow them to do it. So this solution is not that good. - -What we want is to prevent the selection on double-click, that's it. - -A text selection is the default browser action on `mousedown` event. So the alternative solution would be to handle `mousedown` and prevent it, like this: +In this particular case the most reasonable way is to prevent the browser action on `mousedown`. It prevents both these selections: ```html autorun height=50 Before... @@ -189,28 +177,12 @@ Before... ...After ``` -Now the bold element is not selected on double clicks. +Now the bold element is not selected on double clicks, and pressing the left button on it won't start the selection. -The text inside it is still selectable. However, the selection should start not on the text itself, but before or after it. Usually that's fine though. - -````smart header="Canceling the selection" -Instead of *preventing* the selection, we can cancel it "post-factum" in the event handler. - -Here's how: - -```html autorun height=50 -Before... -<b ondblclick="*!*getSelection().removeAllRanges()*/!*"> - Double-click me -</b> -...After -``` - -If you double-click on the bold element, then the selection appears and then is immediately removed. That doesn't look nice though. -```` +Please note: the text inside it is still selectable. However, the selection should start not on the text itself, but before or after it. Usually that's fine for users. ````smart header="Preventing copying" -If we want to disable selection to protect our content from copy-pasting, then we can use another event: `oncopy`. +If we want to disable selection to protect our page content from copy-pasting, then we can use another event: `oncopy`. ```html autorun height=80 no-beautify <div *!*oncopy="alert('Copying forbidden!');return false"*/!*> @@ -221,23 +193,20 @@ If we want to disable selection to protect our content from copy-pasting, then w ``` If you try to copy a piece of text in the `<div>`, that won't work, because the default action `oncopy` is prevented. -Surely that can't stop the user from opening HTML-source, but not everyone knows how to do it. +Surely the user has access to HTML-source of the page, and can take the content from there, but not everyone knows how to do it. ```` ## Summary Mouse events have the following properties: -- Button: `which`. +- Button: `button`. - Modifier keys (`true` if pressed): `altKey`, `ctrlKey`, `shiftKey` and `metaKey` (Mac). - - If you want to handle `key:Ctrl`, then don't forget Mac users, they use `key:Cmd`, so it's better to check `if (e.metaKey || e.ctrlKey)`. + - If you want to handle `key:Ctrl`, then don't forget Mac users, they usually use `key:Cmd`, so it's better to check `if (e.metaKey || e.ctrlKey)`. - Window-relative coordinates: `clientX/clientY`. - Document-relative coordinates: `pageX/pageY`. -It's also important to deal with text selection as an unwanted side-effect of clicks. +The default browser action of `mousedown` is text selection, if it's not good for the interface, then it should be prevented. -There are several ways to do this, for instance: -1. The CSS-property `user-select:none` (with browser prefixes) completely disables text-selection. -2. Cancel the selection post-factum using `getSelection().removeAllRanges()`. -3. Handle `mousedown` and prevent the default action (usually the best). +In the next chapter we'll see more details about events that follow pointer movement and how to track element changes under it. diff --git a/2-ui/3-event-details/1-mouse-events-basics/head.html b/2-ui/3-event-details/1-mouse-events-basics/head.html index 461f0e85b..1b9a73fca 100644 --- a/2-ui/3-event-details/1-mouse-events-basics/head.html +++ b/2-ui/3-event-details/1-mouse-events-basics/head.html @@ -4,25 +4,28 @@ function showmesg(t, form) { - if (timer==0) timer = new Date() + if (timer == 0) { + timer = new Date(); + } + + let tm = new Date(); - let tm = new Date() - if (tm-timer > 300) { - t = '------------------------------\n'+t + if (tm - timer > 300) { + t = '------------------------------\n' + t; } - let area = document.forms[form+'form'].getElementsByTagName('textarea')[0] + let area = document.forms[form + 'form'].getElementsByTagName('textarea')[0]; area.value += t + '\n'; - area.scrollTop = area.scrollHeight + area.scrollTop = area.scrollHeight; - timer = tm + timer = tm; } function logMouse(e) { let evt = e.type; while (evt.length < 11) evt += ' '; - showmesg(evt+" which="+e.which, 'test') + showmesg(evt + " button=" + e.button, 'test') return false; } diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/solution.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/solution.view/index.html index c8e879f2c..84d52b18c 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/solution.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/solution.view/index.html @@ -54,7 +54,7 @@ <p>Once upon a time there was a mother pig who had three little pigs.</p> - <p>The three little pigs grew so big that their mother said to them, "You are too big to live here any longer. You must go and build houses for yourselves. But take care that the wolf does not catch you." + <p>The three little pigs grew so big that their mother said to them, "You are too big to live here any longer. You must go and build houses for yourselves. But take care that the wolf does not catch you."</p> <p>The three little pigs set off. "We will take care that the wolf does not catch us," they said.</p> @@ -78,12 +78,13 @@ } document.onmouseout = function() { - // it is possible that mouseout triggered, but we're still inside the element (cause of bubbling) + // it is possible that mouseout triggered, but we're still inside the element + // (its target was inside, and it bubbled) // but in this case we'll have an immediate mouseover, // so the tooltip will be destroyed and shown again // - // that's an overhead, but here it's not visible - // can be fixed with additional checks + // luckily, the "blinking" won't be visible, + // as both events happen almost at the same time if (tooltip) { tooltip.remove(); tooltip = false; diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/source.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/source.view/index.html index 2dc4394e7..774e24a21 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/source.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/source.view/index.html @@ -54,7 +54,7 @@ <p>Once upon a time there was a mother pig who had three little pigs.</p> - <p>The three little pigs grew so big that their mother said to them, "You are too big to live here any longer. You must go and build houses for yourselves. But take care that the wolf does not catch you." + <p>The three little pigs grew so big that their mother said to them, "You are too big to live here any longer. You must go and build houses for yourselves. But take care that the wolf does not catch you."</p> <p>The three little pigs set off. "We will take care that the wolf does not catch us," they said.</p> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/task.md b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/task.md index 435bac0f8..c77aa0728 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/task.md +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/1-behavior-nested-tooltip/task.md @@ -4,10 +4,12 @@ importance: 5 # Improved tooltip behavior -Write JavaScript that shows a tooltip over an element with the attribute `data-tooltip`. +Write JavaScript that shows a tooltip over an element with the attribute `data-tooltip`. The value of this attribute should become the tooltip text. That's like the task <info:task/behavior-tooltip>, but here the annotated elements can be nested. The most deeply nested tooltip is shown. +Only one tooltip may show up at the same time. + For instance: ```html @@ -21,5 +23,3 @@ For instance: The result in iframe: [iframe src="solution" height=300 border=1] - -P.S. Hint: only one tooltip may show up at the same time. diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.md b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.md index c4af78b11..36eaca1a3 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.md +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.md @@ -3,16 +3,16 @@ The algorithm looks simple: 1. Put `onmouseover/out` handlers on the element. Also can use `onmouseenter/leave` here, but they are less universal, won't work if we introduce delegation. 2. When a mouse cursor entered the element, start measuring the speed on `mousemove`. 3. If the speed is slow, then run `over`. -4. Later if we're out of the element, and `over` was executed, run `out`. +4. When we're going out of the element, and `over` was executed, run `out`. -The question is: "How to measure the speed?" +But how to measure the speed? -The first idea would be: to run our function every `100ms` and measure the distance between previous and new coordinates. If it's small, then the speed is small. +The first idea can be: run a function every `100ms` and measure the distance between previous and new coordinates. If it's small, then the speed is small. Unfortunately, there's no way to get "current mouse coordinates" in JavaScript. There's no function like `getCurrentMouseCoordinates()`. -The only way to get coordinates is to listen to mouse events, like `mousemove`. +The only way to get coordinates is to listen for mouse events, like `mousemove`, and take coordinates from the event object. -So we can set a handler on `mousemove` to track coordinates and remember them. Then we can compare them, once per `100ms`. +So let's set a handler on `mousemove` to track coordinates and remember them. And then compare them, once per `100ms`. P.S. Please note: the solution tests use `dispatchEvent` to see if the tooltip works right. diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/hoverIntent.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/hoverIntent.js index 4e6e2a3e9..7503ca9c2 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/hoverIntent.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/hoverIntent.js @@ -88,7 +88,7 @@ class HoverIntent { if (speed < this.sensitivity) { clearInterval(this.checkSpeedInterval); this.isHover = true; - this.over.call(this.elem, event); + this.over.call(this.elem); } else { // speed fast, remember new coordinates as the previous ones this.prevX = this.lastX; diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/index.html index df0f13656..098232174 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/index.html @@ -20,20 +20,17 @@ <div id="tooltip" hidden>Tooltip</div> <script> - // for the demo - setTimeout(function() { - new HoverIntent({ - elem, - over() { - tooltip.style.left = elem.getBoundingClientRect().left + 5 + 'px'; - tooltip.style.top = elem.getBoundingClientRect().bottom + 5 + 'px'; - tooltip.hidden = false; - }, - out() { - tooltip.hidden = true; - } - }); - }, 2000); + new HoverIntent({ + elem, + over() { + tooltip.style.left = elem.getBoundingClientRect().left + 5 + 'px'; + tooltip.style.top = elem.getBoundingClientRect().bottom + 5 + 'px'; + tooltip.hidden = false; + }, + out() { + tooltip.hidden = true; + } + }); </script> </body> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/test.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/test.js index 7c7f6d23d..f5d4aaffb 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/test.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/solution.view/test.js @@ -49,18 +49,18 @@ describe("hoverIntent", function() { } }) - it("mouseover -> immediately no tooltip", function() { + it("mouseover -> when the pointer just arrived, no tooltip", function() { mouse('mouseover', 10, 10); assert.isFalse(isOver); }); - it("mouseover -> pause shows tooltip", function() { + it("mouseover -> after a delay, the tooltip shows up", function() { mouse('mouseover', 10, 10); this.clock.tick(100); assert.isTrue(isOver); }); - it("mouseover -> fast mouseout no tooltip", function() { + it("mouseover -> followed by fast mouseout leads doesn't show tooltip", function() { mouse('mouseover', 10, 10); setTimeout( () => mouse('mouseout', 300, 300, { relatedTarget: document.body}), diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/hoverIntent.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/hoverIntent.js index caca940b1..a38b42bc2 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/hoverIntent.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/hoverIntent.js @@ -45,6 +45,7 @@ class HoverIntent { destroy() { /* your code to "disable" the functionality, remove all handlers */ + /* it's needed for the tests to work */ } } diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/index.html index 702b79f96..098232174 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/index.html @@ -1,12 +1,11 @@ <!doctype html> -<html lang="en"> +<html> <head> <meta charset="UTF-8"> - <title>Document</title> <link rel="stylesheet" href="style.css"> <script src="hoverIntent.js"></script> - <script src="https://js.cx/test/libs.js"></script> + <script src="https://en.js.cx/test/libs.js"></script> <script src="test.js"></script> </head> @@ -21,18 +20,17 @@ <div id="tooltip" hidden>Tooltip</div> <script> - // for the demo - setTimeout(function() { - new HoverIntent({ - elem, - over() { - tooltip.hidden = false; - }, - out() { - tooltip.hidden = true; - } - }); - }, 2000); + new HoverIntent({ + elem, + over() { + tooltip.style.left = elem.getBoundingClientRect().left + 5 + 'px'; + tooltip.style.top = elem.getBoundingClientRect().bottom + 5 + 'px'; + tooltip.hidden = false; + }, + out() { + tooltip.hidden = true; + } + }); </script> </body> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/style.css index 980e9457e..fa2f09eba 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/style.css @@ -2,6 +2,10 @@ color: red; } +body { + margin: 0; +} + .minutes { color: green; } @@ -20,9 +24,15 @@ top: 0; } -.tooltip { +#tooltip { position: absolute; - background: #eee; - border: 1px brown solid; - padding: 3px; + padding: 10px 20px; + border: 1px solid #b3c9ce; + border-radius: 4px; + text-align: center; + font: italic 14px/1.3 sans-serif; + color: #333; + background: #fff; + z-index: 100000; + box-shadow: 3px 3px 3px rgba(0, 0, 0, .3); } diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/test.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/test.js index 44369d8c3..f5d4aaffb 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/test.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/source.view/test.js @@ -3,7 +3,7 @@ describe("hoverIntent", function() { function mouse(eventType, x, y, options) { - let eventOptions = Object.assign({ + let eventOptions = Object.assign({ bubbles: true, clientX: x, clientY: y, @@ -11,15 +11,15 @@ describe("hoverIntent", function() { pageY: y, target: elem }, options || {}); - + elem.dispatchEvent(new MouseEvent(eventType, eventOptions)); } let isOver; let hoverIntent; - - + + before(function() { this.clock = sinon.useFakeTimers(); }); @@ -27,11 +27,11 @@ describe("hoverIntent", function() { after(function() { this.clock.restore(); }); - - + + beforeEach(function() { isOver = false; - + hoverIntent = new HoverIntent({ elem: elem, over: function() { @@ -49,18 +49,18 @@ describe("hoverIntent", function() { } }) - it("mouseover -> immediately no tooltip", function() { + it("mouseover -> when the pointer just arrived, no tooltip", function() { mouse('mouseover', 10, 10); assert.isFalse(isOver); }); - - it("mouseover -> pause shows tooltip", function() { + + it("mouseover -> after a delay, the tooltip shows up", function() { mouse('mouseover', 10, 10); this.clock.tick(100); assert.isTrue(isOver); }); - it("mouseover -> fast mouseout no tooltip", function() { + it("mouseover -> followed by fast mouseout leads doesn't show tooltip", function() { mouse('mouseover', 10, 10); setTimeout( () => mouse('mouseout', 300, 300, { relatedTarget: document.body}), @@ -94,5 +94,5 @@ describe("hoverIntent", function() { this.clock.tick(200); assert.isFalse(isOver); }); - + }); diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/task.md b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/task.md index 3dfb7e9e1..72e615bdd 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/task.md +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/2-hoverintent/task.md @@ -4,16 +4,17 @@ importance: 5 # "Smart" tooltip -Write a function that shows a tooltip over an element only if the visitor moves the mouse *over it*, but not *through it*. +Write a function that shows a tooltip over an element only if the visitor moves the mouse *to it*, but not *through it*. -In other words, if the visitor moves the mouse on the element and stopped -- show the tooltip. And if they just moved the mouse through fast, then no need, who wants extra blinking? +In other words, if the visitor moves the mouse to the element and stops there -- show the tooltip. And if they just moved the mouse through, then no need, who wants extra blinking? Technically, we can measure the mouse speed over the element, and if it's slow then we assume that it comes "over the element" and show the tooltip, if it's fast -- then we ignore it. -Make a universal object `new HoverIntent(options)` for it. With `options`: +Make a universal object `new HoverIntent(options)` for it. +Its `options`: - `elem` -- element to track. -- `over` -- a function to call if the mouse is slowly moving the element. +- `over` -- a function to call if the mouse came to the element: that is, it moves slowly or stopped over it. - `out` -- a function to call when the mouse leaves the element (if `over` was called). An example of using such object for the tooltip: diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/article.md b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/article.md index ac20f84fa..7997a16bc 100644 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/article.md +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/article.md @@ -1,29 +1,29 @@ -# Moving: mouseover/out, mouseenter/leave +# Moving the mouse: mouseover/out, mouseenter/leave -Let's dive into more details about events that happen when mouse moves between elements. +Let's dive into more details about events that happen when the mouse moves between elements. -## Mouseover/mouseout, relatedTarget +## Events mouseover/mouseout, relatedTarget The `mouseover` event occurs when a mouse pointer comes over an element, and `mouseout` -- when it leaves.  -These events are special, because they have a `relatedTarget`. +These events are special, because they have property `relatedTarget`. This property complements `target`. When a mouse leaves one element for another, one of them becomes `target`, and the other one - `relatedTarget`. For `mouseover`: - `event.target` -- is the element where the mouse came over. -- `event.relatedTarget` -- is the element from which the mouse came. +- `event.relatedTarget` -- is the element from which the mouse came (`relatedTarget` -> `target`). For `mouseout` the reverse: -- `event.target` -- is the element that mouse left. -- `event.relatedTarget` -- is the new under-the-pointer element (that mouse left for). +- `event.target` -- is the element that the mouse left. +- `event.relatedTarget` -- is the new under-the-pointer element, that mouse left for (`target` -> `relatedTarget`). ```online -In the example below each face feature is an element. When you move the mouse, you can see mouse events in the text area. +In the example below each face and its features are separate elements. When you move the mouse, you can see mouse events in the text area. -Each event has the information about where the element came and where it came from. +Each event has the information about both `target` and `relatedTarget`: [codetabs src="mouseoverout" height=280] ``` @@ -36,86 +36,130 @@ That's normal and just means that the mouse came not from another element, but f We should keep that possibility in mind when using `event.relatedTarget` in our code. If we access `event.relatedTarget.tagName`, then there will be an error. ``` -## Events frequency +## Skipping elements The `mousemove` event triggers when the mouse moves. But that doesn't mean that every pixel leads to an event. The browser checks the mouse position from time to time. And if it notices changes then triggers the events. -That means that if the visitor is moving the mouse very fast then DOM-elements may be skipped: +That means that if the visitor is moving the mouse very fast then some DOM-elements may be skipped:  -If the mouse moves very fast from `#FROM` to `#TO` elements as painted above, then intermediate `<div>` (or some of them) may be skipped. The `mouseout` event may trigger on `#FROM` and then immediately `mouseover` on `#TO`. +If the mouse moves very fast from `#FROM` to `#TO` elements as painted above, then intermediate `<div>` elements (or some of them) may be skipped. The `mouseout` event may trigger on `#FROM` and then immediately `mouseover` on `#TO`. -In practice that's helpful, because if there may be many intermediate elements. We don't really want to process in and out of each one. +That's good for performance, because there may be many intermediate elements. We don't really want to process in and out of each one. -On the other hand, we should keep in mind that we can't assume that the mouse slowly moves from one event to another. No, it can "jump". +On the other hand, we should keep in mind that the mouse pointer doesn't "visit" all elements along the way. It can "jump". -In particular it's possible that the cursor jumps right inside the middle of the page from out of the window. And `relatedTarget=null`, because it came from "nowhere": +In particular, it's possible that the pointer jumps right inside the middle of the page from out of the window. In that case `relatedTarget` is `null`, because it came from "nowhere":  +<<<<<<< HEAD <div style="display:none"> In case of a fast move, intermediate elements may trigger no events. But if the mouse enters the element (`mouseover`), when we're guaranteed to have `mouseout` when it leaves it. </div> +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b ```online -Check it out "live" on a teststand below. +You can check it out "live" on a teststand below. -The HTML is two nested `<div>` elements. If you move the mouse fast over them, then there may be no events at all, or maybe only the red div triggers events, or maybe the green one. +Its HTML has two nested elements: the `<div id="child">` is inside the `<div id="parent">`. If you move the mouse fast over them, then maybe only the child div triggers events, or maybe the parent one, or maybe there will be no events at all. -Also try to move the pointer over the red `div`, and then move it out quickly down through the green one. If the movement is fast enough then the parent element is ignored. +Also move the pointer into the child `div`, and then move it out quickly down through the parent one. If the movement is fast enough, then the parent element is ignored. The mouse will cross the parent element without noticing it. [codetabs height=360 src="mouseoverout-fast"] ``` -## "Extra" mouseout when leaving for a child +```smart header="If `mouseover` triggered, there must be `mouseout`" +In case of fast mouse movements, intermediate elements may be ignored, but one thing we know for sure: if the pointer "officially" entered an element (`mouseover` event generated), then upon leaving it we always get `mouseout`. +``` + +## Mouseout when leaving for a child + +<<<<<<< HEAD + +======= +An important feature of `mouseout` -- it triggers, when the pointer moves from an element to its descendant, e.g. from `#parent` to `#child` in this HTML: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b -Imagine -- a mouse pointer entered an element. The `mouseover` triggered. Then the cursor goes into a child element. The interesting fact is that `mouseout` triggers in that case. The cursor is still in the element, but we have a `mouseout` from it! +```html +<div id="parent"> + <div id="child">...</div> +</div> +``` + +If we're on `#parent` and then move the pointer deeper into `#child`, we get `mouseout` on `#parent`!  -That seems strange, but can be easily explained. +That may seem strange, but can be easily explained. -**According to the browser logic, the mouse cursor may be only over a *single* element at any time -- the most nested one (and top by z-index).** +**According to the browser logic, the mouse cursor may be only over a *single* element at any time -- the most nested one and top by z-index.** -So if it goes to another element (even a descendant), then it leaves the previous one. That simple. +So if it goes to another element (even a descendant), then it leaves the previous one. -There's a funny consequence that we can see on the example below. +Please note another important detail of event processing. -The red `<div>` is nested inside the blue one. The blue `<div>` has `mouseover/out` handlers that log all events in the textarea below. +The `mouseover` event on a descendant bubbles up. So, if `#parent` has `mouseover` handler, it triggers: -Try entering the blue element and then moving the mouse on the red one -- and watch the events: + + +```online +You can see that very well in the example below: `<div id="child">` is inside the `<div id="parent">`. There are `mouseover/out` handlers on `#parent` element that output event details. + +If you move the mouse from `#parent` to `#child`, you see two events on `#parent`: +1. `mouseout [target: parent]` (left the parent), then +2. `mouseover [target: child]` (came to the child, bubbled). [codetabs height=360 src="mouseoverout-child"] +``` -1. On entering the blue one -- we get `mouseover [target: blue]`. -2. Then after moving from the blue to the red one -- we get `mouseout [target: blue]` (left the parent). -3. ...And immediately `mouseover [target: red]`. +As shown, when the pointer moves from `#parent` element to `#child`, two handlers trigger on the parent element: `mouseout` and `mouseover`: -So, for a handler that does not take `target` into account, it looks like we left the parent in `mouseout` in `(2)` and returned back to it by `mouseover` in `(3)`. +```js +parent.onmouseout = function(event) { + /* event.target: parent element */ +}; +parent.onmouseover = function(event) { + /* event.target: child element (bubbled) */ +}; +``` -If we perform some actions on entering/leaving the element, then we'll get a lot of extra "false" runs. For simple stuff that may be unnoticeable. For complex things that may bring unwanted side-effects. +**If we don't examine `event.target` inside the handlers, then it may seem that the mouse pointer left `#parent` element, and then immediately came back over it.** -We can fix it by using `mouseenter/mouseleave` events instead. +But that's not the case! The pointer is still over the parent, it just moved deeper into the child element. + +If there are some actions upon leaving the parent element, e.g. an animation runs in `parent.onmouseout`, we usually don't want it when the pointer just goes deeper into `#parent`. + +To avoid it, we can check `relatedTarget` in the handler and, if the mouse is still inside the element, then ignore such event. + +Alternatively we can use other events: `mouseenter` and `mouseleave`, that we'll be covering now, as they don't have such problems. ## Events mouseenter and mouseleave -Events `mouseenter/mouseleave` are like `mouseover/mouseout`. They also trigger when the mouse pointer enters/leaves the element. +Events `mouseenter/mouseleave` are like `mouseover/mouseout`. They trigger when the mouse pointer enters/leaves the element. -But there are two differences: +But there are two important differences: -1. Transitions inside the element are not counted. +1. Transitions inside the element, to/from descendants, are not counted. 2. Events `mouseenter/mouseleave` do not bubble. -These events are intuitively very clear. +These events are extremely simple. -When the pointer enters an element -- the `mouseenter` triggers, and then doesn't matter where it goes while inside the element. The `mouseleave` event only triggers when the cursor leaves it. +When the pointer enters an element -- `mouseenter` triggers. The exact location of the pointer inside the element or its descendants doesn't matter. -If we make the same example, but put `mouseenter/mouseleave` on the blue `<div>`, and do the same -- we can see that events trigger only on entering and leaving the blue `<div>`. No extra events when going to the red one and back. Children are ignored. +When the pointer leaves an element -- `mouseleave` triggers. + +```online +This example is similar to the one above, but now the top element has `mouseenter/mouseleave` instead of `mouseover/mouseout`. + +As you can see, the only generated events are the ones related to moving the pointer in and out of the top element. Nothing happens when the pointer goes to the child and back. Transitions between descendants are ignored [codetabs height=340 src="mouseleave"] +``` ## Event delegation @@ -123,16 +167,16 @@ Events `mouseenter/leave` are very simple and easy to use. But they do not bubbl Imagine we want to handle mouse enter/leave for table cells. And there are hundreds of cells. -The natural solution would be -- to set the handler on `<table>` and process events there. But `mouseenter/leave` don't bubble. So if such event happens on `<td>`, then only a handler on that `<td>` can catch it. +The natural solution would be -- to set the handler on `<table>` and process events there. But `mouseenter/leave` don't bubble. So if such event happens on `<td>`, then only a handler on that `<td>` is able to catch it. -Handlers for `mouseenter/leave` on `<table>` only trigger on entering/leaving the whole table. It's impossible to get any information about transitions inside it. +Handlers for `mouseenter/leave` on `<table>` only trigger when the pointer enters/leaves the table as a whole. It's impossible to get any information about transitions inside it. -Not a problem -- let's use `mouseover/mouseout`. +So, let's use `mouseover/mouseout`. -A simple handler may look like this: +Let's start with simple handlers that highlight the element under mouse: ```js -// let's highlight cells under mouse +// let's highlight an element under the pointer table.onmouseover = function(event) { let target = event.target; target.style.background = 'pink'; @@ -145,41 +189,44 @@ table.onmouseout = function(event) { ``` ```online +Here they are in action. As the mouse travels across the elements of this table, the current one is highlighted: + [codetabs height=480 src="mouseenter-mouseleave-delegation"] ``` -These handlers work when going from any element to any inside the table. - -But we'd like to handle only transitions in and out of `<td>` as a whole. And highlight the cells as a whole. We don't want to handle transitions that happen between the children of `<td>`. +In our case we'd like to handle transitions between table cells `<td>`: entering a cell and leaving it. Other transitions, such as inside the cell or outside of any cells, don't interest us. Let's filter them out. -One of solutions: +Here's what we can do: -- Remember the currently highlighted `<td>` in a variable. +- Remember the currently highlighted `<td>` in a variable, let's call it `currentElem`. - On `mouseover` -- ignore the event if we're still inside the current `<td>`. - On `mouseout` -- ignore if we didn't leave the current `<td>`. -That filters out "extra" events when we are moving between the children of `<td>`. +Here's an example of code that accounts for all possible situations: -```offline -The details are in the [full example](sandbox:mouseenter-mouseleave-delegation-2). -``` +[js src="mouseenter-mouseleave-delegation-2/script.js"] + +Once again, the important features are: +1. It uses event delegation to handle entering/leaving of any `<td>` inside the table. So it relies on `mouseover/out` instead of `mouseenter/leave` that don't bubble and hence allow no delegation. +2. Extra events, such as moving between descendants of `<td>` are filtered out, so that `onEnter/Leave` runs only if the pointer leaves or enters `<td>` as a whole. ```online Here's the full example with all details: -[codetabs height=380 src="mouseenter-mouseleave-delegation-2"] +[codetabs height=460 src="mouseenter-mouseleave-delegation-2"] -Try to move the cursor in and out of table cells and inside them. Fast or slow -- doesn't matter. Only `<td>` as a whole is highlighted unlike the example before. +Try to move the cursor in and out of table cells and inside them. Fast or slow -- doesn't matter. Only `<td>` as a whole is highlighted, unlike the example before. ``` - ## Summary We covered events `mouseover`, `mouseout`, `mousemove`, `mouseenter` and `mouseleave`. -Things that are good to note: +These things are good to note: + +- A fast mouse move may skip intermediate elements. +- Events `mouseover/out` and `mouseenter/leave` have an additional property: `relatedTarget`. That's the element that we are coming from/to, complementary to `target`. + +Events `mouseover/out` trigger even when we go from the parent element to a child element. The browser assumes that the mouse can be only over one element at one time -- the deepest one. -- A fast mouse move can make `mouseover, mousemove, mouseout` to skip intermediate elements. -- Events `mouseover/out` and `mouseenter/leave` have an additional target: `relatedTarget`. That's the element that we are coming from/to, complementary to `target`. -- Events `mouseover/out` trigger even when we go from the parent element to a child element. They assume that the mouse can be only over one element at one time -- the deepest one. -- Events `mouseenter/leave` do not bubble and do not trigger when the mouse goes to a child element. They only track whether the mouse comes inside and outside the element as a whole. +Events `mouseenter/leave` are different in that aspect: they only trigger when the mouse comes in and out the element as a whole. Also they do not bubble. diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/index.html index e129ee6a2..eedd38716 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/index.html @@ -67,6 +67,10 @@ </table> + <textarea id="text"></textarea> + + <input type="button" onclick="text.value=''" value="Clear"> + <script src="script.js"></script> </body> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/script.js index 9f6bf1b5b..6a3202467 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/script.js @@ -2,37 +2,61 @@ let currentElem = null; table.onmouseover = function(event) { - if (currentElem) { - // before entering a new element, the mouse always leaves the previous one - // if we didn't leave <td> yet, then we're still inside it, so can ignore the event - return; - } + // before entering a new element, the mouse always leaves the previous one + // if currentElem is set, we didn't leave the previous <td>, + // that's a mouseover inside it, ignore the event + if (currentElem) return; let target = event.target.closest('td'); - if (!target || !table.contains(target)) return; - // yeah we're inside <td> now + // we moved not into a <td> - ignore + if (!target) return; + + // moved into <td>, but outside of our table (possible in case of nested tables) + // ignore + if (!table.contains(target)) return; + + // hooray! we entered a new <td> currentElem = target; - target.style.background = 'pink'; + onEnter(currentElem); }; table.onmouseout = function(event) { // if we're outside of any <td> now, then ignore the event + // that's probably a move inside the table, but out of <td>, + // e.g. from <tr> to another <tr> if (!currentElem) return; - // we're leaving the element -- where to? Maybe to a child element? + // we're leaving the element – where to? Maybe to a descendant? let relatedTarget = event.relatedTarget; - if (relatedTarget) { // possible: relatedTarget = null - while (relatedTarget) { - // go up the parent chain and check -- if we're still inside currentElem - // then that's an internal transition -- ignore it - if (relatedTarget == currentElem) return; - relatedTarget = relatedTarget.parentNode; - } + + while (relatedTarget) { + // go up the parent chain and check – if we're still inside currentElem + // then that's an internal transition – ignore it + if (relatedTarget == currentElem) return; + + relatedTarget = relatedTarget.parentNode; } - // we left the element. really. - currentElem.style.background = ''; + // we left the <td>. really. + onLeave(currentElem); currentElem = null; }; + +// any functions to handle entering/leaving an element +function onEnter(elem) { + elem.style.background = 'pink'; + + // show that in textarea + text.value += `over -> ${currentElem.tagName}.${currentElem.className}\n`; + text.scrollTop = 1e6; +} + +function onLeave(elem) { + elem.style.background = ''; + + // show that in textarea + text.value += `out <- ${elem.tagName}.${elem.className}\n`; + text.scrollTop = 1e6; +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/style.css index 61e19e363..a5b1fd889 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation-2.view/style.css @@ -16,6 +16,7 @@ vertical-align: bottom; padding-top: 5px; padding-bottom: 12px; + cursor: pointer; } #table .nw { @@ -62,4 +63,4 @@ #table .highlight { background: red; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/script.js index 4700d686e..ae633ad67 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/script.js @@ -1,13 +1,15 @@ table.onmouseover = function(event) { let target = event.target; target.style.background = 'pink'; - text.value += "mouseover " + target.tagName + "\n"; + + text.value += `over -> ${target.tagName}\n`; text.scrollTop = text.scrollHeight; }; table.onmouseout = function(event) { let target = event.target; target.style.background = ''; - text.value += "mouseout " + target.tagName + "\n"; + + text.value += `out <- ${target.tagName}\n`; text.scrollTop = text.scrollHeight; -}; \ No newline at end of file +}; diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/style.css index 61e19e363..a5b1fd889 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseenter-mouseleave-delegation.view/style.css @@ -16,6 +16,7 @@ vertical-align: bottom; padding-top: 5px; padding-bottom: 12px; + cursor: pointer; } #table .nw { @@ -62,4 +63,4 @@ #table .highlight { background: red; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave-table.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave-table.view/style.css index 61e19e363..a5b1fd889 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave-table.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave-table.view/style.css @@ -16,6 +16,7 @@ vertical-align: bottom; padding-top: 5px; padding-bottom: 12px; + cursor: pointer; } #table .nw { @@ -62,4 +63,4 @@ #table .highlight { background: red; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/index.html index 87fc5c635..f26fdbdae 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/index.html @@ -1,15 +1,15 @@ -<!DOCTYPE HTML> +<!doctype html> <html> <head> - <meta charset="utf-8"> + <meta charset="UTF-8"> <link rel="stylesheet" href="style.css"> </head> <body> - <div id="blue" onmouseenter="log(event)" onmouseleave="log(event)"> - <div id="red"></div> + <div id="parent" onmouseenter="mouselog(event)" onmouseleave="mouselog(event)">parent + <div id="child">child</div> </div> <textarea id="text"></textarea> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/script.js index 7b0b70b19..bf64f2de7 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/script.js @@ -1,4 +1,5 @@ -function log(event) { - text.value += event.type + ' [target: ' + event.target.id + ']\n'; +function mouselog(event) { + let d = new Date(); + text.value += `${d.getHours()}:${d.getMinutes()}:${d.getSeconds()} | ${event.type} [target: ${event.target.id}]\n`.replace(/(:|^)(\d\D)/, '$10$2'); text.scrollTop = text.scrollHeight; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/style.css index d4a759838..c4024f32e 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseleave.view/style.css @@ -1,21 +1,22 @@ -#blue { - background: blue; +#parent { + background: #99C0C3; width: 160px; - height: 160px; + height: 120px; position: relative; } -#red { - background: red; - width: 70px; - height: 70px; +#child { + background: #FFDE99; + width: 50%; + height: 50%; position: absolute; - left: 45px; - top: 45px; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); } -#text { +textarea { + height: 140px; + width: 300px; display: block; - height: 100px; - width: 400px; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseover-bubble-nested.svg b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseover-bubble-nested.svg new file mode 100644 index 000000000..6044eff17 --- /dev/null +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseover-bubble-nested.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="258" height="143" viewBox="0 0 258 143"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><filter id="filter-2" width="212.1%" height="203.6%" x="-52.6%" y="-43.3%" filterUnits="objectBoundingBox"><feMorphology in="SourceAlpha" operator="dilate" radius="1" result="shadowSpreadOuter1"/><feOffset dy="1" in="shadowSpreadOuter1" result="shadowOffsetOuter1"/><feGaussianBlur in="shadowOffsetOuter1" result="shadowBlurOuter1" stdDeviation="1.5"/><feComposite in="shadowBlurOuter1" in2="SourceAlpha" operator="out" result="shadowBlurOuter1"/><feColorMatrix in="shadowBlurOuter1" values="0 0 0 0 0.0941176471 0 0 0 0 0.0901960784 0 0 0 0 0.0901960784 0 0 0 1 0"/></filter><path id="path-1" d="M12.487 6.865L0 0l3.64 13.776 2.726-4.108 3.264 4.177 1.474-1.152-3.263-4.177z"/></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="mouseover-bubble-nested.svg"><path id="Rectangle-210" fill="#DBAF88" d="M184.47 57L200 119H78l15.53-62z"/><path id="Rectangle-209" stroke="#91C2A3" stroke-width="18" d="M207.628 30l20.205 66H50.167l20.205-66h137.256z"/><path id="Fill-54" fill="#C06334" d="M109.435 56l-.683 3.884c-15.408-.725-31.056.348-45.752 3.22l5.647 7.733c12.417-2.427 25.638-3.334 38.656-2.721L106.62 72c7.56-2.333 26.38-5.03 26.38-5.03S118.836 59.132 109.435 56z" transform="rotate(33 98 64)"/><g id="Rectangle-237" transform="translate(126 85)"><use fill="#000" filter="url(#filter-2)" xlink:href="#path-1"/><path fill="#FFF" stroke="#181717" d="M-.781-1l14.484 7.963-5.041 1.792 3.144 4.025-2.262 1.767-3.145-4.025-2.958 4.459L-.78-1z"/></g><text id="mouseout" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="39" y="39">mouseout</tspan></text><text id="mouseover" fill="#643B0C" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="144" y="75">mouseover</tspan></text><text id="#parent" fill="#91C2A3" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="4" y="18">#parent</tspan></text><text id="#child" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="4" y="124">#child</tspan></text><path id="Fill-46" fill="#C06334" d="M124.709 37.276a2.456 2.456 0 000 3.452 2.404 2.404 0 003.42 0l5.452-5.499v40.33a2.43 2.43 0 002.42 2.441 2.43 2.43 0 002.418-2.441v-40.33l5.451 5.5a2.405 2.405 0 003.422 0c.944-.954.944-2.5 0-3.453l-9.485-9.569a2.392 2.392 0 00-1.807-.705 2.395 2.395 0 00-1.808.705l-9.483 9.569z"/></g></g></svg> \ No newline at end of file diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/index.html index 20c96c0a7..b110a199d 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/index.html @@ -8,8 +8,8 @@ <body> - <div class="blue" onmouseover="mouselog(event)" onmouseout="mouselog(event)"> - <div class="red"></div> + <div id="parent" onmouseover="mouselog(event)" onmouseout="mouselog(event)">parent + <div id="child">child</div> </div> <textarea id="text"></textarea> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/script.js index 98a098150..bf64f2de7 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/script.js @@ -1,4 +1,5 @@ function mouselog(event) { - text.value += event.type + ' [target: ' + event.target.className + ']\n' - text.scrollTop = text.scrollHeight -} \ No newline at end of file + let d = new Date(); + text.value += `${d.getHours()}:${d.getMinutes()}:${d.getSeconds()} | ${event.type} [target: ${event.target.id}]\n`.replace(/(:|^)(\d\D)/, '$10$2'); + text.scrollTop = text.scrollHeight; +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/style.css index 49e3f9d6d..c4024f32e 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-child.view/style.css @@ -1,21 +1,22 @@ -.blue { - background: blue; +#parent { + background: #99C0C3; width: 160px; - height: 160px; + height: 120px; position: relative; } -.red { - background: red; - width: 100px; - height: 100px; +#child { + background: #FFDE99; + width: 50%; + height: 50%; position: absolute; - left: 30px; - top: 30px; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); } textarea { - height: 100px; - width: 400px; + height: 140px; + width: 300px; display: block; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/index.html b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/index.html index 1e38d693c..9a2691a2f 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/index.html +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/index.html @@ -1,5 +1,5 @@ <!doctype html> -<html lang="en"> +<html> <head> <meta charset="UTF-8"> @@ -8,13 +8,11 @@ <body> - <div id="green"> - <div id="red">Test</div> + <div id="parent">parent + <div id="child">child</div> </div> - - <input onclick="clearText()" value="Clear" type="button"> - <textarea id="text"></textarea> + <input onclick="clearText()" value="Clear" type="button"> <script src="script.js"></script> diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/script.js index 63ec8cdcb..5752e83ae 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/script.js @@ -1,47 +1,48 @@ - green.onmouseover = green.onmouseout = green.onmousemove = handler; +let parent = document.getElementById('parent'); +parent.onmouseover = parent.onmouseout = parent.onmousemove = handler; - function handler(event) { - let type = event.type; - while (type < 11) type += ' '; +function handler(event) { + let type = event.type; + while (type.length < 11) type += ' '; - log(type + " target=" + event.target.id) - return false; - } + log(type + " target=" + event.target.id) + return false; +} - function clearText() { - text.value = ""; - lastMessage = ""; - } +function clearText() { + text.value = ""; + lastMessage = ""; +} - let lastMessageTime = 0; - let lastMessage = ""; - let repeatCounter = 1; +let lastMessageTime = 0; +let lastMessage = ""; +let repeatCounter = 1; - function log(message) { - if (lastMessageTime == 0) lastMessageTime = new Date(); +function log(message) { + if (lastMessageTime == 0) lastMessageTime = new Date(); - let time = new Date(); + let time = new Date(); - if (time - lastMessageTime > 500) { - message = '------------------------------\n' + message; - } + if (time - lastMessageTime > 500) { + message = '------------------------------\n' + message; + } - if (message === lastMessage) { - repeatCounter++; - if (repeatCounter == 2) { - text.value = text.value.trim() + ' x 2\n'; - } else { - text.value = text.value.slice(0, text.value.lastIndexOf('x') + 1) + repeatCounter + "\n"; - } + if (message === lastMessage) { + repeatCounter++; + if (repeatCounter == 2) { + text.value = text.value.trim() + ' x 2\n'; + } else { + text.value = text.value.slice(0, text.value.lastIndexOf('x') + 1) + repeatCounter + "\n"; + } - } else { - repeatCounter = 1; - text.value += message + "\n"; - } + } else { + repeatCounter = 1; + text.value += message + "\n"; + } - text.scrollTop = text.scrollHeight; + text.scrollTop = text.scrollHeight; - lastMessageTime = time; - lastMessage = message; - } \ No newline at end of file + lastMessageTime = time; + lastMessage = message; +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/style.css b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/style.css index e6ae1a2b0..c4024f32e 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/style.css +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout-fast.view/style.css @@ -1,23 +1,22 @@ -#green { - height: 50px; +#parent { + background: #99C0C3; width: 160px; - background: green; + height: 120px; + position: relative; } -#red { - height: 20px; - width: 110px; - background: red; - color: white; - font-weight: bold; - padding: 5px; - text-align: center; - margin: 20px; +#child { + background: #FFDE99; + width: 50%; + height: 50%; + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); } -#text { - font-size: 12px; - height: 200px; - width: 360px; +textarea { + height: 140px; + width: 300px; display: block; -} \ No newline at end of file +} diff --git a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout.view/script.js b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout.view/script.js index 8fa60cc2d..073ee7792 100755 --- a/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout.view/script.js +++ b/2-ui/3-event-details/3-mousemove-mouseover-mouseout-mouseenter-mouseleave/mouseoverout.view/script.js @@ -7,9 +7,9 @@ function handler(event) { return el.className || el.tagName; } - log.value += event.type + ': ' + + log.value += event.type + ': ' + 'target=' + str(event.target) + - ', relatedTarget=' + str(event.relatedTarget) + "\n"; + ', relatedTarget=' + str(event.relatedTarget) + "\n"; log.scrollTop = log.scrollHeight; if (event.type == 'mouseover') { diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/1-slider/solution.md b/2-ui/3-event-details/4-mouse-drag-and-drop/1-slider/solution.md index 28e8cffa9..6d8878d4a 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/1-slider/solution.md +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/1-slider/solution.md @@ -1,4 +1,5 @@ +As we can see from HTML/CSS, the slider is a `<div>` with a colored background, that contains a runner -- another `<div>` with `position:relative`. -We have a horizontal Drag'n'Drop here. +To position the runner we use `position:relative`, to provide the coordinates relative to its parent, here it's more convenient here than `position:absolute`. -To position the element we use `position:relative` and slider-relative coordinates for the thumb. Here it's more convenient here than `position:absolute`. +Then we implement horizontal-only Drag'n'Drop with limitation by width. diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.md b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.md index deb43b655..62cbdb9c5 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.md +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.md @@ -1,5 +1,5 @@ -To drag the element we can use `position:fixed`, it makes coordinates easier to manage. At the end we should switch it back to `position:absolute`. +To drag the element we can use `position:fixed`, it makes coordinates easier to manage. At the end we should switch it back to `position:absolute` to lay the element into the document. -Then, when coordinates are at window top/bottom, we use `window.scrollTo` to scroll it. +When coordinates are at window top/bottom, we use `window.scrollTo` to scroll it. More details in the code, in comments. diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/field.svg b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/field.svg index 4ae90b1c7..17991d792 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/field.svg +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/field.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="233" height="156" viewBox="0 0 233 156"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="field.svg"><image id="Screen-Shot-2017-02-25-at-23.45.22" width="224" height="150" x="4" y="3" opacity=".7" xlink:href=""/><text id="(0,0)" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="45" y="60">(0,0)</tspan></text><circle id="Oval" cx="15.5" cy="15.5" r="4.5" fill="#C06334"/><text id="clientWidth" fill="#A7333A" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold"><tspan x="73" y="114">clientWidth</tspan></text><path id="Line-9" fill="#C06334" fill-rule="nonzero" d="M24.114 22.183l20.711 4.719-5.078 6.181 13.705 11.258 1.16.952-1.905 2.318-1.16-.952-13.704-11.258-5.078 6.183-8.651-19.401z"/><path id="Line-10" fill="#C06334" fill-rule="nonzero" d="M197 118l19 9.5-19 9.5v-8H34v8l-19-9.5 19-9.5v8h163v-8z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/soccer.js b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/soccer.js index 1c1443a7e..10ae2eeed 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/soccer.js +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/solution.view/soccer.js @@ -7,7 +7,7 @@ document.addEventListener('mousedown', function(event) { if (!dragElement) return; event.preventDefault(); - + dragElement.ondragstart = function() { return false; }; @@ -19,7 +19,7 @@ document.addEventListener('mousedown', function(event) { function onMouseUp(event) { finishDrag(); }; - + function onMouseMove(event) { moveAt(event.clientX, event.clientY); } @@ -31,9 +31,9 @@ document.addEventListener('mousedown', function(event) { if(isDragging) { return; } - + isDragging = true; - + document.addEventListener('mousemove', onMouseMove); element.addEventListener('mouseup', onMouseUp); @@ -50,10 +50,10 @@ document.addEventListener('mousedown', function(event) { if(!isDragging) { return; } - + isDragging = false; - dragElement.style.top = parseInt(dragElement.style.top) + pageYOffset + 'px'; + dragElement.style.top = parseInt(dragElement.style.top) + window.pageYOffset + 'px'; dragElement.style.position = 'absolute'; document.removeEventListener('mousemove', onMouseMove); @@ -113,4 +113,4 @@ document.addEventListener('mousedown', function(event) { dragElement.style.top = newY + 'px'; } -}); \ No newline at end of file +}); diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/task.md b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/task.md index 16add0cf8..91fbaa0f2 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/task.md +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/2-drag-heroes/task.md @@ -12,8 +12,8 @@ Requirements: - Use event delegation to track drag start: a single event handler on `document` for `mousedown`. - If elements are dragged to top/bottom window edges -- the page scrolls up/down to allow further dragging. -- There is no horizontal scroll. -- Draggable elements should never leave the window, even after swift mouse moves. +- There is no horizontal scroll (this makes the task a bit simpler, adding it is easy). +- Draggable elements or their parts should never leave the window, even after swift mouse moves. The demo is too big to fit it here, so here's the link. diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/article.md b/2-ui/3-event-details/4-mouse-drag-and-drop/article.md index 6f9238143..4c928eef1 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/article.md +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/article.md @@ -1,40 +1,36 @@ # Drag'n'Drop with mouse events -Drag'n'Drop is a great interface solution. Taking something, dragging and dropping is a clear and simple way to do many things, from copying and moving (see file managers) to ordering (drop into cart). +Drag'n'Drop is a great interface solution. Taking something and dragging and dropping it is a clear and simple way to do many things, from copying and moving documents (as in file managers) to ordering (dropping items into a cart). -In the modern HTML standard there's a [section about Drag Events](https://html.spec.whatwg.org/multipage/interaction.html#dnd). +In the modern HTML standard there's a [section about Drag and Drop](https://html.spec.whatwg.org/multipage/interaction.html#dnd) with special events such as `dragstart`, `dragend`, and so on. -They are interesting, because they allow to solve simple tasks easily, and also allow to handle drag'n'drop of "external" files into the browser. So we can take a file in the OS file-manager and drop it into the browser window. Then JavaScript gains access to its contents. +These events allow us to support special kinds of drag'n'drop, such as handling dragging a file from OS file-manager and dropping it into the browser window. Then JavaScript can access the contents of such files. -But native Drag Events also have limitations. For instance, we can limit dragging by a certain area. Also we can't make it "horizontal" or "vertical" only. There are other drag'n'drop tasks that can't be implemented using that API. +But native Drag Events also have limitations. For instance, we can't prevent dragging from a certain area. Also we can't make the dragging "horizontal" or "vertical" only. And there are many other drag'n'drop tasks that can't be done using them. Also, mobile device support for such events is very weak. -So here we'll see how to implement Drag'n'Drop using mouse events. Not that hard either. +So here we'll see how to implement Drag'n'Drop using mouse events. ## Drag'n'Drop algorithm The basic Drag'n'Drop algorithm looks like this: -1. Catch `mousedown` on a draggable element. -2. Prepare the element for moving (maybe create a copy of it or whatever). -3. Then on `mousemove` move it by changing `left/top` and `position:absolute`. -4. On `mouseup` (button release) -- perform all actions related to a finished Drag'n'Drop. +1. On `mousedown` - prepare the element for moving, if needed (maybe create a clone of it, add a class to it or whatever). +2. Then on `mousemove` move it by changing `left/top` with `position:absolute`. +3. On `mouseup` - perform all actions related to finishing the drag'n'drop. -These are the basics. We can extend it, for instance, by highlighting droppable (available for the drop) elements when hovering over them. +These are the basics. Later we'll see how to add other features, such as highlighting current underlying elements while we drag over them. -Here's the algorithm for drag'n'drop of a ball: +Here's the implementation of dragging a ball: ```js -ball.onmousedown = function(event) { // (1) start the process - - // (2) prepare to moving: make absolute and on top by z-index +ball.onmousedown = function(event) { + // (1) prepare to moving: make absolute and on top by z-index ball.style.position = 'absolute'; ball.style.zIndex = 1000; + // move it out of any current parents directly into body // to make it positioned relative to the body - document.body.append(ball); - // ...and put that absolutely positioned ball under the cursor - - moveAt(event.pageX, event.pageY); + document.body.append(ball); // centers the ball at (pageX, pageY) coordinates function moveAt(pageX, pageY) { @@ -42,14 +38,17 @@ ball.onmousedown = function(event) { // (1) start the process ball.style.top = pageY - ball.offsetHeight / 2 + 'px'; } + // move our absolutely positioned ball under the pointer + moveAt(event.pageX, event.pageY); + function onMouseMove(event) { moveAt(event.pageX, event.pageY); } - // (3) move the ball on mousemove + // (2) move the ball on mousemove document.addEventListener('mousemove', onMouseMove); - // (4) drop the ball, remove unneeded handlers + // (3) drop the ball, remove unneeded handlers ball.onmouseup = function() { document.removeEventListener('mousemove', onMouseMove); ball.onmouseup = null; @@ -65,10 +64,10 @@ Here's an example in action: [iframe src="ball" height=230] -Try to drag'n'drop the mouse and you'll see the strange behavior. +Try to drag'n'drop with the mouse and you'll see such behavior. ``` -That's because the browser has its own Drag'n'Drop for images and some other elements that runs automatically and conflicts with ours. +That's because the browser has its own drag'n'drop support for images and some other elements. It runs automatically and conflicts with ours. To disable it: @@ -88,28 +87,30 @@ In action: Another important aspect -- we track `mousemove` on `document`, not on `ball`. From the first sight it may seem that the mouse is always over the ball, and we can put `mousemove` on it. -But as we remember, `mousemove` triggers often, but not for every pixel. So after swift move the cursor can jump from the ball somewhere in the middle of document (or even outside of the window). +But as we remember, `mousemove` triggers often, but not for every pixel. So after swift move the pointer can jump from the ball somewhere in the middle of document (or even outside of the window). So we should listen on `document` to catch it. ## Correct positioning -In the examples above the ball is always centered under the pointer: +In the examples above the ball is always moved so that its center is under the pointer: ```js ball.style.left = pageX - ball.offsetWidth / 2 + 'px'; ball.style.top = pageY - ball.offsetHeight / 2 + 'px'; ``` -Not bad, but there's a side-effect. To initiate the drag'n'drop can we `mousedown` anywhere on the ball. If do it at the edge, then the ball suddenly "jumps" to become centered. +Not bad, but there's a side effect. To initiate the drag'n'drop, we can `mousedown` anywhere on the ball. But if "take" it from its edge, then the ball suddenly "jumps" to become centered under the mouse pointer. It would be better if we keep the initial shift of the element relative to the pointer. -For instance, if we start dragging by the edge of the ball, then the cursor should remain over the edge while dragging. +For instance, if we start dragging by the edge of the ball, then the pointer should remain over the edge while dragging.  -1. When a visitor presses the button (`mousedown`) -- we can remember the distance from the cursor to the left-upper corner of the ball in variables `shiftX/shiftY`. We should keep that distance while dragging. +Let's update our algorithm: + +1. When a visitor presses the button (`mousedown`) - remember the distance from the pointer to the left-upper corner of the ball in variables `shiftX/shiftY`. We'll keep that distance while dragging. To get these shifts we can substract the coordinates: @@ -119,13 +120,11 @@ For instance, if we start dragging by the edge of the ball, then the cursor shou let shiftY = event.clientY - ball.getBoundingClientRect().top; ``` - Please note that there's no method to get document-relative coordinates in JavaScript, so we use window-relative coordinates here. - 2. Then while dragging we position the ball on the same shift relative to the pointer, like this: ```js // onmousemove - // ball has position:absoute + // ball has position:absolute ball.style.left = event.pageX - *!*shiftX*/!* + 'px'; ball.style.top = event.pageY - *!*shiftY*/!* + 'px'; ``` @@ -146,7 +145,8 @@ ball.onmousedown = function(event) { moveAt(event.pageX, event.pageY); - // centers the ball at (pageX, pageY) coordinates + // moves the ball at (pageX, pageY) coordinates + // taking initial shifts into account function moveAt(pageX, pageY) { ball.style.left = pageX - *!*shiftX*/!* + 'px'; ball.style.top = pageY - *!*shiftY*/!* + 'px'; @@ -156,10 +156,10 @@ ball.onmousedown = function(event) { moveAt(event.pageX, event.pageY); } - // (3) move the ball on mousemove + // move the ball on mousemove document.addEventListener('mousemove', onMouseMove); - // (4) drop the ball, remove unneeded handlers + // drop the ball, remove unneeded handlers ball.onmouseup = function() { document.removeEventListener('mousemove', onMouseMove); ball.onmouseup = null; @@ -178,25 +178,27 @@ In action (inside `<iframe>`): [iframe src="ball3" height=230] ``` -The difference is especially noticeable if we drag the ball by its right-bottom corner. In the previous example the ball "jumps" under the pointer. Now it fluently follows the cursor from the current position. +The difference is especially noticeable if we drag the ball by its right-bottom corner. In the previous example the ball "jumps" under the pointer. Now it fluently follows the pointer from the current position. -## Detecting droppables +## Potential drop targets (droppables) -In previous examples the ball could be dropped just "anywhere" to stay. In real-life we usually take one element and drop it onto another. For instance, a file into a folder, or a user into a trash can or whatever. +In previous examples the ball could be dropped just "anywhere" to stay. In real-life we usually take one element and drop it onto another. For instance, a "file" into a "folder" or something else. -Abstractly, we take a "draggable" element and drop it onto "droppable" element. +Speaking abstract, we take a "draggable" element and drop it onto "droppable" element. -We need to know the target droppable at the end of Drag'n'Drop -- to do the corresponding action, and, preferably, during the dragging process, to highlight it. +We need to know: +- where the element was dropped at the end of Drag'n'Drop -- to do the corresponding action, +- and, preferably, know the droppable we're dragging over, to highlight it. The solution is kind-of interesting and just a little bit tricky, so let's cover it here. -What's the first idea? Probably to put `onmouseover/mouseup` handlers on potential droppables and detect when the mouse pointer appears over them. And then we know that we are dragging/dropping on that element. +What may be the first idea? Probably to set `mouseover/mouseup` handlers on potential droppables? But that doesn't work. The problem is that, while we're dragging, the draggable element is always above other elements. And mouse events only happen on the top element, not on those below it. -For instance, below are two `<div>` elements, red on top of blue. There's no way to catch an event on the blue one, because the red is on top: +For instance, below are two `<div>` elements, red one on top of the blue one (fully covers). There's no way to catch an event on the blue one, because the red is on top: ```html run autorun height=60 <style> @@ -211,32 +213,35 @@ For instance, below are two `<div>` elements, red on top of blue. There's no way <div style="background:red" onmouseover="alert('over red!')"></div> ``` -The same with a draggable element. The ball in always on top over other elements, so events happen on it. Whatever handlers we set on lower elements, they won't work. +The same with a draggable element. The ball is always on top over other elements, so events happen on it. Whatever handlers we set on lower elements, they won't work. That's why the initial idea to put handlers on potential droppables doesn't work in practice. They won't run. So, what to do? -There's a method called `document.elementFromPoint(clientX, clientY)`. It returns the most nested element on given window-relative coordinates (or `null` if coordinates are out of the window). +There's a method called `document.elementFromPoint(clientX, clientY)`. It returns the most nested element on given window-relative coordinates (or `null` if given coordinates are out of the window). If there are multiple overlapping elements on the same coordinates, then the topmost one is returned. -So in any of our mouse event handlers we can detect the potential droppable under the pointer like this: +We can use it in any of our mouse event handlers to detect the potential droppable under the pointer, like this: ```js // in a mouse event handler -ball.hidden = true; // (*) +ball.hidden = true; // (*) hide the element that we drag + let elemBelow = document.elementFromPoint(event.clientX, event.clientY); +// elemBelow is the element below the ball, may be droppable + ball.hidden = false; -// elemBelow is the element below the ball. If it's droppable, we can handle it. ``` -Please note: we need to hide the ball before the call `(*)`. Otherwise we'll usually have a ball on these coordinates, as it's the top element under the pointer: `elemBelow=ball`. +Please note: we need to hide the ball before the call `(*)`. Otherwise we'll usually have a ball on these coordinates, as it's the top element under the pointer: `elemBelow=ball`. So we hide it and immediately show again. -We can use that code to check what we're "flying over" at any time. And handle the drop when it happens. +We can use that code to check what element we're "flying over" at any time. And handle the drop when it happens. An extended code of `onMouseMove` to find "droppable" elements: ```js -let currentDroppable = null; // potential droppable that we're flying over right now +// potential droppable that we're flying over right now +let currentDroppable = null; function onMouseMove(event) { moveAt(event.pageX, event.pageY); @@ -246,16 +251,16 @@ function onMouseMove(event) { ball.hidden = false; // mousemove events may trigger out of the window (when the ball is dragged off-screen) - // if clientX/clientY are out of the window, then elementfromPoint returns null + // if clientX/clientY are out of the window, then elementFromPoint returns null if (!elemBelow) return; // potential droppables are labeled with the class "droppable" (can be other logic) let droppableBelow = elemBelow.closest('.droppable'); - if (currentDroppable != droppableBelow) { // if there are any changes + if (currentDroppable != droppableBelow) { // we're flying in or out... // note: both values can be null - // currentDroppable=null if we were not over a droppable (e.g over an empty space) + // currentDroppable=null if we were not over a droppable before this event (e.g over an empty space) // droppableBelow=null if we're not over a droppable now, during this event if (currentDroppable) { @@ -271,28 +276,28 @@ function onMouseMove(event) { } ``` -In the example below when the ball is dragged over the soccer gate, the gate is highlighted. +In the example below when the ball is dragged over the soccer goal, the goal is highlighted. [codetabs height=250 src="ball4"] -Now we have the current "drop target" in the variable `currentDroppable` during the whole process and can use it to highlight or any other stuff. +Now we have the current "drop target", that we're flying over, in the variable `currentDroppable` during the whole process and can use it to highlight or any other stuff. ## Summary -We considered a basic `Drag'n'Drop` algorithm. +We considered a basic Drag'n'Drop algorithm. The key components: -1. Events flow: `ball.mousedown` -> `document.mousemove` -> `ball.mouseup` (cancel native `ondragstart`). +1. Events flow: `ball.mousedown` -> `document.mousemove` -> `ball.mouseup` (don't forget to cancel native `ondragstart`). 2. At the drag start -- remember the initial shift of the pointer relative to the element: `shiftX/shiftY` and keep it during the dragging. 3. Detect droppable elements under the pointer using `document.elementFromPoint`. We can lay a lot on this foundation. -- On `mouseup` we can finalize the drop: change data, move elements around. +- On `mouseup` we can intellectually finalize the drop: change data, move elements around. - We can highlight the elements we're flying over. - We can limit dragging by a certain area or direction. - We can use event delegation for `mousedown/up`. A large-area event handler that checks `event.target` can manage Drag'n'Drop for hundreds of elements. - And so on. -There are frameworks that build architecture over it: `DragZone`, `Droppable`, `Draggable` and other classes. Most of them do the similar stuff to described above, so it should be easy to understand them now. Or roll our own, because you already know how to handle the process, and it may be more flexible than to adapt something else. +There are frameworks that build architecture over it: `DragZone`, `Droppable`, `Draggable` and other classes. Most of them do the similar stuff to what's described above, so it should be easy to understand them now. Or roll your own, as you can see that that's easy enough to do, sometimes easier than adapting a third-party solution. diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/ball.view/index.html b/2-ui/3-event-details/4-mouse-drag-and-drop/ball.view/index.html index 3fdd7fe76..8751c70ad 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/ball.view/index.html +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/ball.view/index.html @@ -13,16 +13,13 @@ <script> - ball.onmousedown = function(event) { // (1) start the process - - // (2) prepare to moving: make absolute and top by z-index + ball.onmousedown = function(event) { ball.style.position = 'absolute'; ball.style.zIndex = 1000; document.body.appendChild(ball); - // ...and put that absolutely positioned ball under the cursor + moveAt(event.pageX, event.pageY); - // centers the ball at (pageX, pageY) coordinates function moveAt(pageX, pageY) { ball.style.left = pageX - ball.offsetWidth / 2 + 'px'; ball.style.top = pageY - ball.offsetHeight / 2 + 'px'; @@ -32,10 +29,8 @@ moveAt(event.pageX, event.pageY); } - // (3) move the ball on mousemove document.addEventListener('mousemove', onMouseMove); - // (4) drop the ball, remove unneeded handlers ball.onmouseup = function() { document.removeEventListener('mousemove', onMouseMove); ball.onmouseup = null; diff --git a/2-ui/3-event-details/4-mouse-drag-and-drop/ball2.view/index.html b/2-ui/3-event-details/4-mouse-drag-and-drop/ball2.view/index.html index 3843aa789..195107e56 100644 --- a/2-ui/3-event-details/4-mouse-drag-and-drop/ball2.view/index.html +++ b/2-ui/3-event-details/4-mouse-drag-and-drop/ball2.view/index.html @@ -13,16 +13,13 @@ <script> - ball.onmousedown = function(event) { // (1) start the process - - // (2) prepare to moving: make absolute and top by z-index + ball.onmousedown = function(event) { ball.style.position = 'absolute'; ball.style.zIndex = 1000; document.body.appendChild(ball); - // ...and put that absolutely positioned ball under the cursor + moveAt(event.pageX, event.pageY); - // centers the ball at (pageX, pageY) coordinates function moveAt(pageX, pageY) { ball.style.left = pageX - ball.offsetWidth / 2 + 'px'; ball.style.top = pageY - ball.offsetHeight / 2 + 'px'; @@ -32,10 +29,8 @@ moveAt(event.pageX, event.pageY); } - // (3) move the ball on mousemove document.addEventListener('mousemove', onMouseMove); - // (4) drop the ball, remove unneeded handlers ball.onmouseup = function() { document.removeEventListener('mousemove', onMouseMove); ball.onmouseup = null; diff --git a/2-ui/3-event-details/6-pointer-events/article.md b/2-ui/3-event-details/6-pointer-events/article.md new file mode 100644 index 000000000..ecc144712 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/article.md @@ -0,0 +1,282 @@ +# Pointer events + +Pointer events are a modern way to handle input from a variety of pointing devices, such as a mouse, a pen/stylus, a touchscreen, and so on. + +## The brief history + +Let's make a small overview, so that you understand the general picture and the place of Pointer Events among other event types. + +- Long ago, in the past, there were only mouse events. + + Then touch devices became widespread, phones and tablets in particular. For the existing scripts to work, they generated (and still generate) mouse events. For instance, tapping a touchscreen generates `mousedown`. So touch devices worked well with web pages. + + But touch devices have more capabilities than a mouse. For example, it's possible to touch multiple points at once ("multi-touch"). Although, mouse events don't have necessary properties to handle such multi-touches. + +- So touch events were introduced, such as `touchstart`, `touchend`, `touchmove`, that have touch-specific properties (we don't cover them in detail here, because pointer events are even better). + + Still, it wasn't enough, as there are many other devices, such as pens, that have their own features. Also, writing code that listens for both touch and mouse events was cumbersome. + +- To solve these issues, the new standard Pointer Events was introduced. It provides a single set of events for all kinds of pointing devices. + +As of now, [Pointer Events Level 2](https://www.w3.org/TR/pointerevents2/) specification is supported in all major browsers, while the newer [Pointer Events Level 3](https://w3c.github.io/pointerevents/) is in the works and is mostly compatible with Pointer Events level 2. + +Unless you develop for old browsers, such as Internet Explorer 10, or for Safari 12 or below, there's no point in using mouse or touch events any more -- we can switch to pointer events. + +Then your code will work well with both touch and mouse devices. + +That said, there are some important peculiarities that one should know in order to use Pointer Events correctly and avoid surprises. We'll make note of them in this article. + +## Pointer event types + +Pointer events are named similarly to mouse events: + +| Pointer event | Similar mouse event | +|---------------|-------------| +| `pointerdown` | `mousedown` | +| `pointerup` | `mouseup` | +| `pointermove` | `mousemove` | +| `pointerover` | `mouseover` | +| `pointerout` | `mouseout` | +| `pointerenter` | `mouseenter` | +| `pointerleave` | `mouseleave` | +| `pointercancel` | - | +| `gotpointercapture` | - | +| `lostpointercapture` | - | + +As we can see, for every `mouse<event>`, there's a `pointer<event>` that plays a similar role. Also there are 3 additional pointer events that don't have a corresponding `mouse...` counterpart, we'll explain them soon. + +```smart header="Replacing `mouse<event>` with `pointer<event>` in our code" +We can replace `mouse<event>` events with `pointer<event>` in our code and expect things to continue working fine with mouse. + +The support for touch devices will also "magically" improve. Although, we may need to add `touch-action: none` in some places in CSS. We'll cover it below in the section about `pointercancel`. +``` + +## Pointer event properties + +Pointer events have the same properties as mouse events, such as `clientX/Y`, `target`, etc., plus some others: + +- `pointerId` - the unique identifier of the pointer causing the event. + + Browser-generated. Allows us to handle multiple pointers, such as a touchscreen with stylus and multi-touch (examples will follow). +- `pointerType` - the pointing device type. Must be a string, one of: "mouse", "pen" or "touch". + + We can use this property to react differently on various pointer types. +- `isPrimary` - is `true` for the primary pointer (the first finger in multi-touch). + +Some pointer devices measure contact area and pressure, e.g. for a finger on the touchscreen, there are additional properties for that: + +- `width` - the width of the area where the pointer (e.g. a finger) touches the device. Where unsupported, e.g. for a mouse, it's always `1`. +- `height` - the height of the area where the pointer touches the device. Where unsupported, it's always `1`. +- `pressure` - the pressure of the pointer tip, in range from 0 to 1. For devices that don't support pressure must be either `0.5` (pressed) or `0`. +- `tangentialPressure` - the normalized tangential pressure. +- `tiltX`, `tiltY`, `twist` - pen-specific properties that describe how the pen is positioned relative to the surface. + +These properties aren't supported by most devices, so they are rarely used. You can find the details about them in the [specification](https://w3c.github.io/pointerevents/#pointerevent-interface) if needed. + +## Multi-touch + +One of the things that mouse events totally don't support is multi-touch: a user can touch in several places at once on their phone or tablet, or perform special gestures. + +Pointer Events allow handling multi-touch with the help of the `pointerId` and `isPrimary` properties. + +Here's what happens when a user touches a touchscreen in one place, then puts another finger somewhere else on it: + +1. At the first finger touch: + - `pointerdown` with `isPrimary=true` and some `pointerId`. +2. For the second finger and more fingers (assuming the first one is still touching): + - `pointerdown` with `isPrimary=false` and a different `pointerId` for every finger. + +Please note: the `pointerId` is assigned not to the whole device, but for each touching finger. If we use 5 fingers to simultaneously touch the screen, we have 5 `pointerdown` events, each with their respective coordinates and a different `pointerId`. + +The events associated with the first finger always have `isPrimary=true`. + +We can track multiple touching fingers using their `pointerId`. When the user moves and then removes a finger, we get `pointermove` and `pointerup` events with the same `pointerId` as we had in `pointerdown`. + +```online +Here's the demo that logs `pointerdown` and `pointerup` events: + +[iframe src="multitouch" edit height=200] + +Please note: you must be using a touchscreen device, such as a phone or a tablet, to actually see the difference in `pointerId/isPrimary`. For single-touch devices, such as a mouse, there'll be always same `pointerId` with `isPrimary=true`, for all pointer events. +``` + +## Event: pointercancel + +The `pointercancel` event fires when there's an ongoing pointer interaction, and then something happens that causes it to be aborted, so that no more pointer events are generated. + +Such causes are: +- The pointer device hardware was physically disabled. +- The device orientation changed (tablet rotated). +- The browser decided to handle the interaction on its own, considering it a mouse gesture or zoom-and-pan action or something else. + +We'll demonstrate `pointercancel` on a practical example to see how it affects us. + +Let's say we're implementing drag'n'drop for a ball, just as in the beginning of the article <info:mouse-drag-and-drop>. + +Here is the flow of user actions and the corresponding events: + +1) The user presses on an image, to start dragging + - `pointerdown` event fires +2) Then they start moving the pointer (thus dragging the image) + - `pointermove` fires, maybe several times +3) And then the surprise happens! The browser has native drag'n'drop support for images, that kicks in and takes over the drag'n'drop process, thus generating `pointercancel` event. + - The browser now handles drag'n'drop of the image on its own. The user may even drag the ball image out of the browser, into their Mail program or a File Manager. + - No more `pointermove` events for us. + +So the issue is that the browser "hijacks" the interaction: `pointercancel` fires in the beginning of the "drag-and-drop" process, and no more `pointermove` events are generated. + +```online +Here's the drag'n'drop demo with logging of pointer events (only `up/down`, `move` and `cancel`) in the `textarea`: + +[iframe src="ball" height=240 edit] +``` + +We'd like to implement the drag'n'drop on our own, so let's tell the browser not to take it over. + +**Prevent the default browser action to avoid `pointercancel`.** + +We need to do two things: + +1. Prevent native drag'n'drop from happening: + - We can do this by setting `ball.ondragstart = () => false`, just as described in the article <info:mouse-drag-and-drop>. + - That works well for mouse events. +2. For touch devices, there are other touch-related browser actions (besides drag'n'drop). To avoid problems with them too: + - Prevent them by setting `#ball { touch-action: none }` in CSS. + - Then our code will start working on touch devices. + +After we do that, the events will work as intended, the browser won't hijack the process and doesn't emit `pointercancel`. + +```online +This demo adds these lines: + +[iframe src="ball-2" height=240 edit] + +As you can see, there's no `pointercancel` any more. +``` + +Now we can add the code to actually move the ball, and our drag'n'drop will work for mouse devices and touch devices. + +## Pointer capturing + +Pointer capturing is a special feature of pointer events. + +The idea is very simple, but may seem quite odd at first, as nothing like that exists for any other event type. + +The main method is: +- `elem.setPointerCapture(pointerId)` -- binds events with the given `pointerId` to `elem`. After the call all pointer events with the same `pointerId` will have `elem` as the target (as if happened on `elem`), no matter where in document they really happened. + +In other words, `elem.setPointerCapture(pointerId)` retargets all subsequent events with the given `pointerId` to `elem`. + +The binding is removed: +- automatically when `pointerup` or `pointercancel` events occur, +- automatically when `elem` is removed from the document, +- when `elem.releasePointerCapture(pointerId)` is called. + +Now what is it good for? It's time to see a real-life example. + +**Pointer capturing can be used to simplify drag'n'drop kind of interactions.** + +Let's recall how one can implement a custom slider, described in the <info:mouse-drag-and-drop>. + +We can make a `slider` element to represent the strip and the "runner" (`thumb`) inside it: + +```html +<div class="slider"> + <div class="thumb"></div> +</div> +``` + +With styles, it looks like this: + +[iframe src="slider-html" height=40 edit] + +<p></p> + +And here's the working logic, as it was described, after replacing mouse events with similar pointer events: + +1. The user presses on the slider `thumb` -- `pointerdown` triggers. +2. Then they move the pointer -- `pointermove` triggers, and our code moves the `thumb` element along. + - ...As the pointer moves, it may leave the slider `thumb` element, go above or below it. The `thumb` should move strictly horizontally, remaining aligned with the pointer. + +In the mouse event based solution, to track all pointer movements, including when it goes above/below the `thumb`, we had to assign `mousemove` event handler on the whole `document`. + +That's not a cleanest solution, though. One of the problems is that when a user moves the pointer around the document, it may trigger event handlers (such as `mouseover`) on some other elements, invoke totally unrelated UI functionality, and we don't want that. + +This is the place where `setPointerCapture` comes into play. + +- We can call `thumb.setPointerCapture(event.pointerId)` in `pointerdown` handler, +- Then future pointer events until `pointerup/cancel` will be retargeted to `thumb`. +- When `pointerup` happens (dragging complete), the binding is removed automatically, we don't need to care about it. + +So, even if the user moves the pointer around the whole document, events handlers will be called on `thumb`. Nevertheless, coordinate properties of the event objects, such as `clientX/clientY` will still be correct - the capturing only affects `target/currentTarget`. + +Here's the essential code: + +```js +thumb.onpointerdown = function(event) { + // retarget all pointer events (until pointerup) to thumb + thumb.setPointerCapture(event.pointerId); + + // start tracking pointer moves + thumb.onpointermove = function(event) { + // moving the slider: listen on the thumb, as all pointer events are retargeted to it + let newLeft = event.clientX - slider.getBoundingClientRect().left; + thumb.style.left = newLeft + 'px'; + }; + + // on pointer up finish tracking pointer moves + thumb.onpointerup = function(event) { + thumb.onpointermove = null; + thumb.onpointerup = null; + // ...also process the "drag end" if needed + }; +}; + +// note: no need to call thumb.releasePointerCapture, +// it happens on pointerup automatically +``` + +```online +The full demo: + +[iframe src="slider" height=100 edit] + +<p></p> + +In the demo, there's also an additional element with `onmouseover` handler showing the current date. + +Please note: while you're dragging the thumb, you may hover over this element, and its handler *does not* trigger. + +So the dragging is now free of side effects, thanks to `setPointerCapture`. +``` + + + +At the end, pointer capturing gives us two benefits: +1. The code becomes cleaner as we don't need to add/remove handlers on the whole `document` any more. The binding is released automatically. +2. If there are other pointer event handlers in the document, they won't be accidentally triggered by the pointer while the user is dragging the slider. + +### Pointer capturing events + +There's one more thing to mention here, for the sake of completeness. + +There are two events associated with pointer capturing: + +- `gotpointercapture` fires when an element uses `setPointerCapture` to enable capturing. +- `lostpointercapture` fires when the capture is released: either explicitly with `releasePointerCapture` call, or automatically on `pointerup`/`pointercancel`. + +## Summary + +Pointer events allow handling mouse, touch and pen events simultaneously, with a single piece of code. + +Pointer events extend mouse events. We can replace `mouse` with `pointer` in event names and expect our code to continue working for mouse, with better support for other device types. + +For drag'n'drops and complex touch interactions that the browser may decide to hijack and handle on its own - remember to cancel the default action on events and set `touch-action: none` in CSS for elements that we engage. + +Additional abilities of pointer events are: + +- Multi-touch support using `pointerId` and `isPrimary`. +- Device-specific properties, such as `pressure`, `width/height`, and others. +- Pointer capturing: we can retarget all pointer events to a specific element until `pointerup`/`pointercancel`. + +As of now, pointer events are supported in all major browsers, so we can safely switch to them, especially if IE10- and Safari 12- are not needed. And even with those browsers, there are polyfills that enable the support of pointer events. diff --git a/2-ui/3-event-details/6-pointer-events/ball-2.view/index.html b/2-ui/3-event-details/6-pointer-events/ball-2.view/index.html new file mode 100644 index 000000000..5f3abbcb0 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/ball-2.view/index.html @@ -0,0 +1,38 @@ +<!doctype html> +<body style="height: 200px"> + <style> + #ball { + touch-action: none; + } + </style> + + <p>Drag the ball.</p> + + <img src="https://js.cx/clipart/ball.svg" style="cursor:pointer" width="40" height="40" id="ball"> + + <script> + + ball.onpointerdown = log; + ball.onpointerup = log; + ball.onpointermove = log; + ball.onpointercancel = log; + + ball.ondragstart = () => false; + + let lastEventType; + let n = 1; + function log(event) { + if (lastEventType == event.type) { + n++; + text.value = text.value.replace(/.*\n$/, `${event.type} * ${n}\n`); + return; + } + lastEventType = event.type; + n = 1; + text.value += event.type + '\n'; + text.scrollTop = 1e9; + } + </script> + + <textarea id="text" style="display:block;width:300px;height:100px"></textarea> +</body> diff --git a/2-ui/3-event-details/6-pointer-events/ball.view/index.html b/2-ui/3-event-details/6-pointer-events/ball.view/index.html new file mode 100644 index 000000000..8bbef8f63 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/ball.view/index.html @@ -0,0 +1,30 @@ +<!doctype html> +<body style="height: 200px"> + <p>Drag the ball.</p> + + <img src="https://js.cx/clipart/ball.svg" style="cursor:pointer" width="40" height="40" id="ball"> + + <script> + + ball.onpointerdown = log; + ball.onpointerup = log; + ball.onpointermove = log; + ball.onpointercancel = log; + + let lastEventType; + let n = 1; + function log(event) { + if (lastEventType == event.type) { + n++; + text.value = text.value.replace(/.*\n$/, `${event.type} * ${n}\n`); + return; + } + lastEventType = event.type; + n = 1; + text.value += event.type + '\n'; + text.scrollTop = 1e9; + } + </script> + + <textarea id="text" style="display:block;width:300px;height:100px"></textarea> +</body> diff --git a/2-ui/3-event-details/6-pointer-events/multitouch.view/index.html b/2-ui/3-event-details/6-pointer-events/multitouch.view/index.html new file mode 100644 index 000000000..d46e1bc16 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/multitouch.view/index.html @@ -0,0 +1,28 @@ +<!doctype html> +<body> + <style> + #area { + border: 1px solid black; + width: 90%; + height: 180px; + cursor: pointer; + overflow: scroll; + user-select: none; + } + </style> + + <div id="area"> + Multi-touch here + </div> + <script> + document.onpointerdown = document.onpointerup = log; + + function log(event) { + area.insertAdjacentHTML("beforeend", ` + <div>${event.type} isPrimary=${event.isPrimary} pointerId=${event.pointerId}</div> + `) + area.scrollTop = 1e9; + } + </script> + +</body> diff --git a/2-ui/3-event-details/6-pointer-events/slider-html.view/index.html b/2-ui/3-event-details/6-pointer-events/slider-html.view/index.html new file mode 100644 index 000000000..781016f52 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/slider-html.view/index.html @@ -0,0 +1,6 @@ +<!DOCTYPE html> +<link rel="stylesheet" href="style.css"> + +<div id="slider" class="slider"> + <div class="thumb"></div> +</div> diff --git a/2-ui/3-event-details/6-pointer-events/slider-html.view/style.css b/2-ui/3-event-details/6-pointer-events/slider-html.view/style.css new file mode 100644 index 000000000..9b3d3b82d --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/slider-html.view/style.css @@ -0,0 +1,19 @@ +.slider { + border-radius: 5px; + background: #E0E0E0; + background: linear-gradient(left top, #E0E0E0, #EEEEEE); + width: 310px; + height: 15px; + margin: 5px; +} + +.thumb { + width: 10px; + height: 25px; + border-radius: 3px; + position: relative; + left: 10px; + top: -5px; + background: blue; + cursor: pointer; +} diff --git a/2-ui/3-event-details/6-pointer-events/slider.view/index.html b/2-ui/3-event-details/6-pointer-events/slider.view/index.html new file mode 100644 index 000000000..b29e646a1 --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/slider.view/index.html @@ -0,0 +1,50 @@ +<!DOCTYPE html> +<link rel="stylesheet" href="style.css"> + +<div id="slider" class="slider"> + <div class="thumb"></div> +</div> + +<p style="border:1px solid gray" onmousemove="this.textContent = new Date()">Mouse over here to see the date</p> + +<script> + let thumb = slider.querySelector('.thumb'); + let shiftX; + + function onThumbDown(event) { + event.preventDefault(); // prevent selection start (browser action) + + shiftX = event.clientX - thumb.getBoundingClientRect().left; + + thumb.setPointerCapture(event.pointerId); + + thumb.onpointermove = onThumbMove; + + thumb.onpointerup = event => { + // dragging finished, no need to track pointer any more + // ...any other "drag end" logic here... + thumb.onpointermove = null; + thumb.onpointerup = null; + } + }; + + function onThumbMove(event) { + let newLeft = event.clientX - shiftX - slider.getBoundingClientRect().left; + + // if the pointer is out of slider => adjust left to be within the boundaries + if (newLeft < 0) { + newLeft = 0; + } + let rightEdge = slider.offsetWidth - thumb.offsetWidth; + if (newLeft > rightEdge) { + newLeft = rightEdge; + } + + thumb.style.left = newLeft + 'px'; + }; + + thumb.onpointerdown = onThumbDown; + + thumb.ondragstart = () => false; + +</script> diff --git a/2-ui/3-event-details/6-pointer-events/slider.view/style.css b/2-ui/3-event-details/6-pointer-events/slider.view/style.css new file mode 100644 index 000000000..a84cd5e7e --- /dev/null +++ b/2-ui/3-event-details/6-pointer-events/slider.view/style.css @@ -0,0 +1,20 @@ +.slider { + border-radius: 5px; + background: #E0E0E0; + background: linear-gradient(left top, #E0E0E0, #EEEEEE); + width: 310px; + height: 15px; + margin: 5px; +} + +.thumb { + touch-action: none; + width: 10px; + height: 25px; + border-radius: 3px; + position: relative; + left: 10px; + top: -5px; + background: blue; + cursor: pointer; +} diff --git a/2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/solution.md b/2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/solution.md similarity index 76% rename from 2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/solution.md rename to 2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/solution.md index 32dc6091b..453f8c946 100644 --- a/2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/solution.md +++ b/2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/solution.md @@ -1,6 +1,6 @@ We should use two handlers: `document.onkeydown` and `document.onkeyup`. -The set `pressed` should keep currently pressed keys. +Let's create a set `pressed = new Set()` to keep currently pressed keys. The first handler adds to it, while the second one removes from it. Every time on `keydown` we check if we have enough keys pressed, and run the function if it is so. diff --git a/2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/solution.view/index.html b/2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/solution.view/index.html similarity index 100% rename from 2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/solution.view/index.html rename to 2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/solution.view/index.html diff --git a/2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/task.md b/2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/task.md similarity index 100% rename from 2-ui/3-event-details/5-keyboard-events/2-check-sync-keydown/task.md rename to 2-ui/3-event-details/7-keyboard-events/2-check-sync-keydown/task.md diff --git a/2-ui/3-event-details/5-keyboard-events/article.md b/2-ui/3-event-details/7-keyboard-events/article.md similarity index 51% rename from 2-ui/3-event-details/5-keyboard-events/article.md rename to 2-ui/3-event-details/7-keyboard-events/article.md index 7decd91be..12fe63201 100644 --- a/2-ui/3-event-details/5-keyboard-events/article.md +++ b/2-ui/3-event-details/7-keyboard-events/article.md @@ -2,7 +2,7 @@ Before we get to keyboard, please note that on modern devices there are other ways to "input something". For instance, people use speech recognition (especially on mobile devices) or copy/paste with the mouse. -So if we want to track any input into an `<input>` field, then keyboard events are not enough. There's another event named `input` to handle changes of an `<input>` field, by any means. And it may be a better choice for such task. We'll cover it later in the chapter <info:events-change-input>. +So if we want to track any input into an `<input>` field, then keyboard events are not enough. There's another event named `input` to track changes of an `<input>` field, by any means. And it may be a better choice for such task. We'll cover it later in the chapter <info:events-change-input>. Keyboard events should be used when we want to handle keyboard actions (virtual keyboard also counts). For instance, to react on arrow keys `key:Up` and `key:Down` or hotkeys (including combinations of keys). @@ -30,7 +30,7 @@ The `keydown` events happens when a key is pressed down, and then `keyup` -- whe The `key` property of the event object allows to get the character, while the `code` property of the event object allows to get the "physical key code". -For instance, the same key `key:Z` can be pressed with or without `Shift`. That gives us two different characters: lowercase `z` and uppercase `Z`. +For instance, the same key `key:Z` can be pressed with or without `key:Shift`. That gives us two different characters: lowercase `z` and uppercase `Z`. The `event.key` is exactly the character, and it will be different. But `event.code` is the same: @@ -52,7 +52,7 @@ For instance: There are several widespread keyboard layouts, and the specification gives key codes for each of them. -See [alphanumeric section of the spec](https://www.w3.org/TR/uievents-code/#key-alphanumeric-section) for more codes, or just try the [teststand](#keyboard-test-stand) above. +Read the [alphanumeric section of the spec](https://www.w3.org/TR/uievents-code/#key-alphanumeric-section) for more codes, or just press a key in the [teststand](#keyboard-test-stand) above. ``` ```warn header="Case matters: `\"KeyZ\"`, not `\"keyZ\"`" @@ -61,9 +61,7 @@ Seems obvious, but people still make mistakes. Please evade mistypes: it's `KeyZ`, not `keyZ`. The check like `event.code=="keyZ"` won't work: the first letter of `"Key"` must be uppercase. ``` - -What if a key does not give any character? For instance, `key:Shift` or `key:F1` or others. For those keys `event.key` is approximately the same as `event.code`: - +What if a key does not give any character? For instance, `key:Shift` or `key:F1` or others. For those keys, `event.key` is approximately the same as `event.code`: | Key | `event.key` | `event.code` | |--------------|-------------|--------------| @@ -73,15 +71,13 @@ What if a key does not give any character? For instance, `key:Shift` or `key:F1` Please note that `event.code` specifies exactly which key is pressed. For instance, most keyboards have two `key:Shift` keys: on the left and on the right side. The `event.code` tells us exactly which one was pressed, and `event.key` is responsible for the "meaning" of the key: what it is (a "Shift"). -Let's say, we want to handle a hotkey: `key:Ctrl+Z` (or `key:Cmd+Z` for Mac). Most text editors hook the "Undo" action on it. We can set a listener on `keydown` and check which key is pressed -- to detect when we have the hotkey. - -Please answer the question -- in such a listener, should we check the value of `event.key` or `event.code`? +Let's say, we want to handle a hotkey: `key:Ctrl+Z` (or `key:Cmd+Z` for Mac). Most text editors hook the "Undo" action on it. We can set a listener on `keydown` and check which key is pressed. -Please, pause and answer. +There's a dilemma here: in such a listener, should we check the value of `event.key` or `event.code`? -Made up your mind? +On one hand, the value of `event.key` is a character, it changes depending on the language. If the visitor has several languages in OS and switches between them, the same key gives different characters. So it makes sense to check `event.code`, it's always the same. -If you've got an understanding, then the answer is, of course, `event.code`, as we don't want `event.key` there. The value of `event.key` can change depending on the language or `CapsLock` enabled. The value of `event.code` is strictly bound to the key, so here we go: +Like this: ```js run document.addEventListener('keydown', function(event) { @@ -91,11 +87,37 @@ document.addEventListener('keydown', function(event) { }); ``` +On the other hand, there's a problem with `event.code`. For different keyboard layouts, the same key may have different characters. + +For example, here are US layout ("QWERTY") and German layout ("QWERTZ") under it (from Wikipedia): + + + + + +For the same key, US layout has "Z", while German layout has "Y" (letters are swapped). + +Literally, `event.code` will equal `KeyZ` for people with German layout when they press `key:Y`. + +If we check `event.code == 'KeyZ'` in our code, then for people with German layout such test will pass when they press `key:Y`. + +That sounds really odd, but so it is. The [specification](https://www.w3.org/TR/uievents-code/#table-key-code-alphanumeric-writing-system) explicitly mentions such behavior. + +So, `event.code` may match a wrong character for unexpected layout. Same letters in different layouts may map to different physical keys, leading to different codes. Luckily, that happens only with several codes, e.g. `keyA`, `keyQ`, `keyZ` (as we've seen), and doesn't happen with special keys such as `Shift`. You can find the list in the [specification](https://www.w3.org/TR/uievents-code/#table-key-code-alphanumeric-writing-system). + +To reliably track layout-dependent characters, `event.key` may be a better way. + +On the other hand, `event.code` has the benefit of staying always the same, bound to the physical key location. So hotkeys that rely on it work well even in case of a language switch. + +Do we want to handle layout-dependant keys? Then `event.key` is the way to go. + +Or we want a hotkey to work even after a language switch? Then `event.code` may be better. + ## Auto-repeat -If a key is being pressed for a long enough time, it starts to repeat: the `keydown` triggers again and again, and then when it's released we finally get `keyup`. So it's kind of normal to have many `keydown` and a single `keyup`. +If a key is being pressed for a long enough time, it starts to "auto-repeat": the `keydown` triggers again and again, and then when it's released we finally get `keyup`. So it's kind of normal to have many `keydown` and a single `keyup`. -For all repeating keys the event object has `event.repeat` property set to `true`. +For events triggered by auto-repeat, the event object has `event.repeat` property set to `true`. ## Default actions @@ -117,22 +139,25 @@ For instance, the `<input>` below expects a phone number, so it does not accept ```html autorun height=60 run <script> function checkPhoneKey(key) { - return (key >= '0' && key <= '9') || key == '+' || key == '(' || key == ')' || key == '-'; + return (key >= '0' && key <= '9') || ['+','(',')','-'].includes(key); } </script> <input *!*onkeydown="return checkPhoneKey(event.key)"*/!* placeholder="Phone, please" type="tel"> ``` -Please note that special keys like `key:Backspace`, `key:Left`, `key:Right`, `key:Ctrl+V` do not work in the input. That's a side-effect of the strict filter `checkPhoneKey`. +The `onkeydown` handler here uses `checkPhoneKey` to check for the key pressed. If it's valid (from `0..9` or one of `+-()`), then it returns `true`, otherwise `false`. + +As we know, the `false` value returned from the event handler, assigned using a DOM property or an attribute, such as above, prevents the default action, so nothing appears in the `<input>` for keys that don't pass the test. (The `true` value returned doesn't affect anything, only returning `false` matters) -Let's relax it a little bit: +Please note that special keys, such as `key:Backspace`, `key:Left`, `key:Right`, do not work in the input. That's a side effect of the strict filter `checkPhoneKey`. These keys make it return `false`. +Let's relax the filter a little bit by allowing arrow keys `key:Left`, `key:Right` and `key:Delete`, `key:Backspace`: ```html autorun height=60 run <script> function checkPhoneKey(key) { - return (key >= '0' && key <= '9') || key == '+' || key == '(' || key == ')' || key == '-' || - key == 'ArrowLeft' || key == 'ArrowRight' || key == 'Delete' || key == 'Backspace'; + return (key >= '0' && key <= '9') || + ['+','(',')','-',*!*'ArrowLeft','ArrowRight','Delete','Backspace'*/!*].includes(key); } </script> <input onkeydown="return checkPhoneKey(event.key)" placeholder="Phone, please" type="tel"> @@ -140,16 +165,21 @@ function checkPhoneKey(key) { Now arrows and deletion works well. -...But we still can enter anything by using a mouse and right-click + Paste. So the filter is not 100% reliable. We can just let it be like that, because most of time it works. Or an alternative approach would be to track the `input` event -- it triggers after any modification. There we can check the new value and highlight/modify it when it's invalid. +Even though we have the key filter, one still can enter anything using a mouse and right-click + Paste. Mobile devices provide other means to enter values. So the filter is not 100% reliable. + +The alternative approach would be to track the `oninput` event -- it triggers *after* any modification. There we can check the new `input.value` and modify it/highlight the `<input>` when it's invalid. Or we can use both event handlers together. ## Legacy In the past, there was a `keypress` event, and also `keyCode`, `charCode`, `which` properties of the event object. -There were so many browser incompatibilities that developers of the specification decided to deprecate all of them. The old code still works, as the browser keep supporting them, but there's totally no need to use those any more. +There were so many browser incompatibilities while working with them, that developers of the specification had no way, other than deprecating all of them and creating new, modern events (described above in this chapter). The old code still works, as browsers keep supporting them, but there's totally no need to use those any more. + +## Mobile Keyboards -There was time when this chapter included their detailed description. But as of now we can forget about those. +When using virtual/mobile keyboards, formally known as IME (Input-Method Editor), the W3C standard states that a KeyboardEvent's [`e.keyCode` should be `229`](https://www.w3.org/TR/uievents/#determine-keydown-keyup-keyCode) and [`e.key` should be `"Unidentified"`](https://www.w3.org/TR/uievents-key/#key-attr-values). +While some of these keyboards might still use the right values for `e.key`, `e.code`, `e.keyCode`... when pressing certain keys such as arrows or backspace, there's no guarantee, so your keyboard logic might not always work on mobile devices. ## Summary @@ -163,8 +193,8 @@ Keyboard events: Main keyboard event properties: - `code` -- the "key code" (`"KeyA"`, `"ArrowLeft"` and so on), specific to the physical location of the key on keyboard. -- `key` -- the character (`"A"`, `"a"` and so on), for non-character keys usually has the same value as `code`. +- `key` -- the character (`"A"`, `"a"` and so on), for non-character keys, such as `key:Esc`, usually has the same value as `code`. -In the past, keyboard events were sometimes used to track user input in form fields. That's not reliable, because the input can come from various sources. We have `input` and `change` events to handle any input (covered later in the chapter <info:events-change-input>). They trigger after any input, including mouse or speech recognition. +In the past, keyboard events were sometimes used to track user input in form fields. That's not reliable, because the input can come from various sources. We have `input` and `change` events to handle any input (covered later in the chapter <info:events-change-input>). They trigger after any kind of input, including copy-pasting or speech recognition. We should use keyboard events when we really want keyboard. For example, to react on hotkeys or special keys. diff --git a/2-ui/3-event-details/7-keyboard-events/german-layout.svg b/2-ui/3-event-details/7-keyboard-events/german-layout.svg new file mode 100644 index 000000000..7ac9a4008 --- /dev/null +++ b/2-ui/3-event-details/7-keyboard-events/german-layout.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="600" height="200" viewBox="0 0 600 200"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="Group" transform="translate(.918 .953)"><path id="rect2186" fill="#FFF" stroke="#181717" d="M.08.083h39.933V39.46H.08z"/><path id="rect2218" fill="#FFF" stroke="#181717" d="M40.013.083h39.934V39.46H40.013z"/><path id="rect2222" fill="#FFF" stroke="#181717" d="M79.947.083h39.933V39.46H79.947z"/><path id="rect2228" fill="#FFF" stroke="#181717" d="M118.882.083h39.933V39.46h-39.933z"/><path id="rect2230" fill="#FFF" stroke="#181717" d="M158.815.083h39.934V39.46h-39.934z"/><path id="rect2232" fill="#FFF" stroke="#181717" d="M198.749.083h39.933V39.46H198.75z"/><path id="rect2234" fill="#FFF" stroke="#181717" d="M239.68.083h39.934V39.46h-39.933z"/><path id="rect2236" fill="#FFF" stroke="#181717" d="M279.614.083h39.933V39.46h-39.933z"/><path id="rect2238" fill="#FFF" stroke="#181717" d="M318.55.083h39.933V39.46h-39.934z"/><path id="rect2240" fill="#FFF" stroke="#181717" d="M358.483.083h39.933V39.46h-39.933z"/><path id="rect2242" fill="#FFF" stroke="#181717" d="M398.416.083h39.933V39.46h-39.933z"/><path id="rect2244" fill="#FFF" stroke="#181717" d="M438.35.083h39.933V39.46h-39.934z"/><path id="rect2246" fill="#FFF" stroke="#181717" d="M478.283.083h39.933V39.46h-39.933z"/><path id="rect2248" fill="#D1CFCD" stroke="#181717" d="M517.218.083h81.864V39.46h-81.864z"/><path id="rect2250" fill="#D1CFCD" stroke="#181717" d="M.08 39.461h59.9V78.84H.08z"/><path id="rect2252" fill="#FFF" stroke="#181717" d="M59.98 39.461h39.933V78.84H59.98z"/><path id="rect2254" fill="#FFF" stroke="#181717" d="M98.915 39.461h39.934V78.84H98.915z"/><path id="rect2256" fill="#FFF" stroke="#181717" d="M138.849 39.461h39.933V78.84h-39.933z"/><path id="rect2258" fill="#FFF" stroke="#181717" d="M178.782 39.461h39.933V78.84h-39.933z"/><path id="rect2262" fill="#FFF" stroke="#181717" d="M219.714 39.461h39.933V78.84h-39.933z"/><path id="rect2264" fill="#FFF" stroke="#181717" d="M259.647 39.461h39.934V78.84h-39.934z"/><path id="rect2266" fill="#FFF" stroke="#181717" d="M298.582 39.461h39.934V78.84h-39.934z"/><path id="rect2270" fill="#FFF" stroke="#181717" d="M338.516 39.461h39.933V78.84h-39.933z"/><path id="rect2272" fill="#FFF" stroke="#181717" d="M378.45 39.461h39.933V78.84h-39.934z"/><path id="rect2274" fill="#FFF" stroke="#181717" d="M418.383 39.461h39.933V78.84h-39.933z"/><path id="rect2278" fill="#FFF" stroke="#181717" d="M458.316 39.461h39.934V78.84h-39.934z"/><path id="rect2280" fill="#FFF" stroke="#181717" d="M497.251 39.461h39.934V78.84H497.25z"/><path id="rect2286" fill="#D1CFCD" stroke="#181717" d="M.08 79.876h69.883v39.378H.08z"/><path id="rect2292" fill="#FFF" stroke="#181717" d="M69.963 79.876h39.934v39.378H69.963z"/><path id="rect2296" fill="#FFF" stroke="#181717" d="M109.897 79.876h39.933v39.378h-39.933z"/><path id="rect2298" fill="#FFF" stroke="#181717" d="M149.83 79.876h39.934v39.378H149.83z"/><path id="rect2300" fill="#FFF" stroke="#181717" d="M188.765 79.876H228.7v39.378h-39.934z"/><path id="rect2302" fill="#FFF" stroke="#181717" d="M228.699 79.876h39.933v39.378H228.7z"/><path id="rect2306" fill="#FFF" stroke="#181717" d="M268.632 79.876h39.934v39.378h-39.934z"/><path id="rect2308" fill="#FFF" stroke="#181717" d="M308.566 79.876h39.933v39.378h-39.933z"/><path id="rect2312" fill="#FFF" stroke="#181717" d="M348.5 79.876h39.933v39.378h-39.934z"/><path id="rect2314" fill="#FFF" stroke="#181717" d="M388.433 79.876h39.933v39.378h-39.933z"/><path id="rect2316" fill="#FFF" stroke="#181717" d="M428.366 79.876H468.3v39.378h-39.934z"/><path id="rect2318" fill="#FFF" stroke="#181717" d="M468.3 79.876h39.933v39.378H468.3z"/><path id="rect2320" fill="#D1CFCD" stroke="#181717" stroke-width="1" d="M547.5 79.358h-10.315V39.46h61.896v79.793h-51.656z"/><path id="rect2322" fill="#D1CFCD" stroke="#181717" stroke-width="1" d="M.08 119.254h49.917v39.378H.08z"/><path id="rect2324" fill="#FFF" stroke="#181717" d="M89.93 119.254h39.934v39.378H89.93z"/><path id="rect2326" fill="#FFF" stroke="#181717" d="M129.864 119.254h39.933v39.378h-39.933z"/><path id="rect2330" fill="#FFF" stroke="#181717" d="M169.797 119.254h39.933v39.378h-39.933z"/><path id="rect2334" fill="#FFF" stroke="#181717" d="M208.732 119.254h39.934v39.378h-39.934z"/><path id="rect2336" fill="#FFF" stroke="#181717" d="M248.666 119.254h39.933v39.378h-39.933z"/><path id="rect2338" fill="#FFF" stroke="#181717" d="M288.599 119.254h39.933v39.378H288.6z"/><path id="rect2340" fill="#FFF" stroke="#181717" d="M328.532 119.254h39.934v39.378h-39.934z"/><path id="rect2342" fill="#FFF" stroke="#181717" d="M368.466 119.254h39.933v39.378h-39.933z"/><path id="rect2344" fill="#FFF" stroke="#181717" d="M408.4 119.254h39.933v39.378h-39.934z"/><path id="rect2346" fill="#FFF" stroke="#181717" d="M448.333 119.254h39.933v39.378h-39.933z"/><path id="rect2348" fill="#D1CFCD" stroke="#181717" d="M488.266 119.254h110.816v39.378H488.266z"/><path id="rect2350" fill="#D1CFCD" stroke="#181717" d="M.08 158.632h59.9v39.378H.08z"/><path id="rect2352" fill="#D1CFCD" stroke="#181717" d="M537.185 158.632h61.897v39.378h-61.897z"/><path id="rect2354" fill="#D1CFCD" stroke="#181717" d="M98.915 158.632h59.9v39.378h-59.9z"/><path id="rect2360" fill="#D1CFCD" stroke="#181717" d="M398.416 158.632h59.9v39.378h-59.9z"/><path id="rect2362" fill="#FFF" stroke="#181717" d="M158.815 158.632h239.601v39.378h-239.6z"/><path id="rect2364" fill="#F6F3F1" stroke="#181717" d="M59.98 158.632h39.933v39.378H59.98z"/><path id="rect2366" fill="#F6F3F1" stroke="#181717" d="M458.316 158.632h39.934v39.378h-39.934z"/><path id="rect2368" fill="#F6F3F1" stroke="#181717" d="M497.251 158.632h39.934v39.378H497.25z"/><path id="text5091" fill="#181717" fill-rule="nonzero" d="M49.737 12.695l-.623-5.28V4.227h2.463v3.186l-.624 5.281h-1.216m-.591 3.266v-2.35h2.375v2.35h-2.375m.591 20.391v-7.798h-2.735v-1.6h.176c.997 0 1.722-.155 2.176-.463.458-.308.727-.823.807-1.545h1.832v11.406h-2.256"/><path id="text5127" fill="#181717" fill-rule="nonzero" d="M89.635 5.264h1.507v4.347h-1.507V5.264m-2.56 0h1.507v4.347h-1.507V5.264m-2.137 31.088c.01-.995.226-1.837.644-2.527.42-.694 1.163-1.397 2.23-2.107.163-.109.399-.26.707-.451 1.414-.897 2.12-1.76 2.12-2.59 0-.492-.148-.88-.447-1.166-.298-.285-.707-.428-1.225-.428-.565 0-1.002.177-1.311.53-.304.347-.456.844-.456 1.492v.093h-2.073c0-1.275.345-2.257 1.037-2.947.69-.689 1.675-1.034 2.952-1.034 1.157 0 2.081.314 2.773.941.69.622 1.036 1.452 1.036 2.488 0 .742-.18 1.385-.542 1.929-.361.544-1.06 1.164-2.097 1.858-.22.15-.523.348-.91.591-.985.628-1.54 1.104-1.665 1.431h5.104v1.897h-7.877"/><path id="text5139" fill="#181717" fill-rule="nonzero" d="M126.869 31.62h2.136c.005.62.155 1.087.45 1.402.296.31.734.465 1.314.465.559 0 .989-.144 1.29-.434.305-.294.458-.715.458-1.263 0-.547-.179-.96-.536-1.24-.353-.284-.878-.426-1.577-.426-.037 0-.094.003-.171.008a2.506 2.506 0 01-.164.008v-1.535h.241c.643 0 1.116-.124 1.422-.372.31-.253.466-.638.466-1.154 0-.429-.132-.767-.396-1.015-.259-.254-.614-.38-1.064-.38-.492 0-.873.145-1.142.434-.27.29-.404.7-.404 1.232v.093h-2.098c.026-1.137.36-2.007 1.002-2.612.648-.61 1.56-.914 2.735-.914 1.113 0 1.996.271 2.65.814.652.542.978 1.273.978 2.193 0 .485-.111.912-.334 1.278-.223.367-.557.67-1.002.907.57.248 1 .592 1.29 1.03.295.435.442.954.442 1.558 0 1.11-.365 1.992-1.095 2.643-.73.65-1.728.976-2.991.976-1.249 0-2.212-.315-2.89-.945-.674-.636-1.01-1.537-1.01-2.705v-.046"/><path id="text5145" fill="#181717" fill-rule="nonzero" d="M168.895 14.91c.468-.057.825-.21 1.07-.46.246-.25.369-.584.369-1 0-.37-.117-.674-.351-.914-.229-.244-.591-.434-1.088-.57v2.944m-1.02-5.341V6.914c-.445.052-.785.193-1.019.422-.234.224-.35.526-.35.906 0 .317.108.583.325.796.222.213.57.39 1.045.531m0 8.55V16.62c-1.291-.073-2.256-.414-2.896-1.023-.64-.609-.959-1.489-.959-2.64h2.296c.028.563.17 1.008.428 1.336.263.323.64.529 1.13.617v-3.233c-1.324-.307-2.266-.708-2.826-1.202-.56-.495-.84-1.171-.84-2.03 0-.953.332-1.72.994-2.304.668-.583 1.559-.887 2.673-.914v-.999h1.019v1c1.113.052 1.981.359 2.604.92.622.558.965 1.344 1.027 2.36h-2.33c-.04-.485-.168-.855-.385-1.11-.211-.26-.517-.406-.916-.437v2.842c1.41.401 2.404.86 2.98 1.375.577.515.866 1.194.866 2.038 0 .994-.337 1.788-1.011 2.381-.668.594-1.613.94-2.835 1.039v1.483h-1.02m1.157 14.095v-4.787l-3.332 4.787h3.332m-.052 4.138V33.94h-5.173v-1.967l4.522-6.481h3.041v6.668h1.422v1.78h-1.422v2.413h-2.39"/><path id="text5151" fill="#181717" fill-rule="nonzero" d="M212.846 12.99c0 .356.11.656.33.9.22.239.493.358.816.358.314 0 .583-.122.809-.366.225-.25.338-.547.338-.892 0-.34-.113-.63-.338-.868a1.048 1.048 0 00-.809-.367c-.323 0-.595.12-.816.359-.22.233-.33.525-.33.876m-1.595 0c0-.829.264-1.527.793-2.095a2.558 2.558 0 011.948-.852c.755 0 1.396.29 1.926.868.534.573.8 1.266.8 2.079 0 .818-.266 1.519-.8 2.103-.534.579-1.176.868-1.926.868-.77 0-1.418-.287-1.948-.86-.529-.574-.793-1.277-.793-2.111m-4.27 2.971l6.276-11.733h1.183l-6.276 11.733h-1.184m-.646-8.762c0 .35.107.645.323.884.22.24.492.359.816.359.318 0 .588-.12.808-.359.226-.244.338-.539.338-.884 0-.34-.112-.63-.338-.868a1.048 1.048 0 00-.808-.367c-.324 0-.595.12-.816.359a1.243 1.243 0 00-.323.876m-1.595 0c0-.828.264-1.53.794-2.103.529-.579 1.175-.868 1.94-.868.754 0 1.396.292 1.925.876.535.579.802 1.277.802 2.095 0 .818-.267 1.516-.802 2.095-.529.573-1.17.86-1.925.86-.77 0-1.419-.284-1.948-.852-.524-.568-.786-1.27-.786-2.103m2.102 25.76h2.058c.034.5.188.887.463 1.163.274.27.642.406 1.102.406.569 0 1.002-.18 1.301-.541.304-.361.456-.88.456-1.554 0-.632-.155-1.128-.463-1.49-.309-.366-.735-.549-1.28-.549-.303 0-.57.067-.8.2-.23.132-.427.329-.588.589l-1.896-.12.683-6.117h6.049v1.928h-4.564l-.272 2.286c.23-.197.495-.343.794-.438.298-.101.632-.152 1-.152 1.048 0 1.893.345 2.535 1.036.647.69.97 1.595.97 2.716 0 1.227-.355 2.206-1.066 2.94-.71.727-1.663 1.09-2.859 1.09-1.088 0-1.955-.3-2.602-.9-.641-.605-.982-1.436-1.021-2.493"/><path id="path3073" fill="#181717" fill-rule="nonzero" d="M248.136 10.477l-.576.414c-.232.16-.406.354-.523.58-.116.226-.174.48-.174.76 0 .427.146.786.44 1.077.293.291.66.437 1.099.437.323 0 .631-.066.925-.196.298-.136.573-.331.826-.587l-2.017-2.485m.607-2.372l.265-.188c.243-.165.422-.351.538-.557.117-.206.175-.442.175-.708 0-.256-.081-.459-.243-.61-.156-.15-.371-.225-.644-.225-.278 0-.498.077-.66.233-.161.15-.242.356-.242.617 0 .14.038.294.113.46.081.165.195.341.342.527l.356.451m-1.706 1.002l-.44-.572a3.576 3.576 0 01-.492-.844 2.442 2.442 0 01-.144-.835c0-.803.258-1.44.773-1.913.52-.477 1.226-.715 2.115-.715.854 0 1.529.218 2.024.655.495.437.743 1.026.743 1.77 0 .562-.147 1.066-.44 1.513-.288.441-.743.858-1.364 1.25l1.6 1.957c.15-.246.27-.522.355-.828.091-.306.155-.648.19-1.024h1.97a6.483 6.483 0 01-.408 1.814 5.604 5.604 0 01-.895 1.529l2.032 2.454h-2.578l-.758-.926a4.741 4.741 0 01-1.47.926 4.572 4.572 0 01-1.668.309c-1.031 0-1.875-.304-2.532-.911-.652-.607-.978-1.383-.978-2.327 0-.702.182-1.307.546-1.814.364-.507.97-.996 1.82-1.468"/><path id="text5163" fill="#181717" fill-rule="nonzero" d="M288.564 36.352h-2.353c.084-1.6.454-3.187 1.11-4.76.66-1.574 1.602-3.12 2.824-4.64h-5.54v-1.999h7.987v1.762c-1.238 1.42-2.19 2.923-2.856 4.507-.661 1.585-1.052 3.295-1.172 5.13"/><path id="path3094" fill="#181717" fill-rule="nonzero" d="M506.735 48.674l-1.417 2.151-1.368-1.032 1.546-2.091-2.255-.677.524-1.761 2.165.832v-2.49h1.594v2.49l2.166-.815.54 1.761-2.255.66 1.53 2.073-1.4 1.085-1.37-2.186"/><path id="path3092" fill="#181717" fill-rule="nonzero" d="M325.79 31.784c0 .541.15.959.45 1.252.304.294.738.44 1.302.44.537 0 .959-.149 1.264-.447.31-.3.465-.714.465-1.245 0-.51-.158-.92-.473-1.228-.315-.31-.734-.464-1.256-.464s-.946.157-1.272.471c-.32.315-.48.722-.48 1.221m.178-4.745c0 .428.134.758.403.99.269.226.654.34 1.155.34.502 0 .887-.116 1.156-.348.274-.232.41-.56.41-.982 0-.412-.139-.736-.418-.973-.28-.237-.662-.356-1.148-.356-.475 0-.855.121-1.14.363-.279.242-.418.564-.418.966m-.799 2.149c-.465-.258-.801-.56-1.008-.905-.207-.345-.31-.785-.31-1.321 0-.948.326-1.692.977-2.233.651-.541 1.55-.812 2.698-.812 1.158 0 2.063.27 2.714.812.657.535.985 1.28.985 2.233 0 .51-.111.953-.333 1.33-.223.37-.556.669-1 .896.542.252.95.602 1.224 1.05.274.444.411.985.411 1.624 0 1.076-.356 1.921-1.07 2.534-.708.614-1.685.92-2.93.92-1.262 0-2.242-.304-2.94-.912-.697-.608-1.046-1.455-1.046-2.542 0-.634.129-1.165.387-1.592.264-.433.678-.794 1.241-1.082"/><path id="path3112" fill="#181717" fill-rule="nonzero" d="M330.53 5.264c-.81 1.15-1.412 2.32-1.806 3.51a11.87 11.87 0 00-.59 3.748c0 1.302.197 2.549.59 3.74.394 1.19.995 2.36 1.805 3.51h-1.857c-1.028-1.216-1.808-2.437-2.342-3.663-.528-1.231-.793-2.427-.793-3.587 0-1.16.265-2.356.793-3.587.534-1.231 1.314-2.455 2.342-3.67h1.857"/><path id="path3110" fill="#181717" fill-rule="nonzero" d="M369.808 27.766c0-.634-.18-1.134-.537-1.5-.353-.37-.835-.556-1.447-.556-.578 0-1.028.178-1.352.533-.318.356-.476.853-.476 1.492 0 .597.167 1.064.502 1.399.335.334.806.502 1.412.502.595 0 1.06-.165 1.395-.495.335-.33.503-.788.503-1.375m.043 2.882c-.289.258-.63.451-1.022.58a4.243 4.243 0 01-1.326.193c-1.219 0-2.195-.327-2.928-.981-.734-.655-1.1-1.53-1.1-2.628 0-1.164.398-2.105 1.195-2.82.797-.717 1.848-1.075 3.154-1.075 1.513 0 2.663.466 3.448 1.399.792.927 1.187 2.282 1.187 4.065 0 1.916-.421 3.385-1.265 4.405-.843 1.02-2.056 1.53-3.639 1.53-1.092 0-1.972-.252-2.642-.757-.67-.51-1.037-1.203-1.1-2.08l2.503.009c.058.355.214.62.468.796.254.175.612.262 1.074.262.624 0 1.1-.242 1.43-.726.33-.484.517-1.208.563-2.172"/><path id="path3121" fill="#181717" fill-rule="nonzero" d="M364.473 5.264h1.875c1.02 1.211 1.796 2.432 2.324 3.663.528 1.227.792 2.425.792 3.595 0 1.165-.264 2.363-.792 3.594-.528 1.232-1.303 2.45-2.324 3.656h-1.875c.815-1.155 1.42-2.328 1.813-3.518.4-1.196.599-2.44.599-3.732 0-1.297-.2-2.541-.599-3.732-.393-1.19-.998-2.366-1.813-3.526"/><path id="path3119" fill="#181717" fill-rule="nonzero" d="M405.986 29.62c0 1.349.15 2.328.451 2.937.307.61.793.915 1.459.915.665 0 1.152-.308 1.458-.922.307-.615.46-1.592.46-2.93 0-1.343-.153-2.32-.46-2.929-.306-.61-.793-.914-1.458-.914-.666 0-1.152.305-1.459.914-.3.605-.451 1.581-.451 2.93m-2.578 0c0-1.907.37-3.333 1.11-4.278.747-.95 1.873-1.426 3.378-1.426 1.499 0 2.622.478 3.368 1.434.752.95 1.129 2.379 1.129 4.285 0 1.906-.374 3.33-1.12 4.27-.741.94-1.867 1.41-3.377 1.41-1.505 0-2.63-.473-3.377-1.418-.74-.945-1.111-2.371-1.111-4.278"/><path id="text5187" fill="#181717" fill-rule="nonzero" d="M453.324 136.87h7.987v1.58h-7.987v-1.58m1.802 14.508v-2.122h4.383v2.122h-4.383"/><path id="path2908" fill="#181717" fill-rule="nonzero" d="M506.976 65.368h1.515v4.328h3.735v1.69h-3.735v4.345h-1.515v-4.346h-3.735v-1.689h3.735v-4.328"/><path id="path2906" fill="#181717" fill-rule="nonzero" d="M403.408 12.951h8.985v1.64h-8.985v-1.64m0-3.542h8.985v1.64h-8.985v-1.64"/><g id="g3468" transform="translate(522.21 13.554)"><path id="path5439" stroke="#181717" stroke-width="3" d="M34.942 5.448H6.61"/><path id="path5441" fill="#181717" d="M10.219 10.79V.105L.329 5.448z"/></g><g id="g5459" transform="translate(10.063 48.788)"><path id="path5453" stroke="#181717" stroke-width="3" d="M25.624 5.699H6.669"/><path id="path5455" fill="#181717" d="M10.649 11.043V.356L.666 5.7z"/><path id="path5457" stroke="#181717" stroke-width="3" d="M.666.356v10.687"/></g><g id="g5464" transform="rotate(-180 18.01 35.275)"><path id="path5466" stroke="#181717" stroke-width="3" d="M25.624 5.699H6.669"/><path id="path5468" fill="#181717" d="M10.649 11.043V.356L.666 5.7z"/><path id="path5470" stroke="#181717" stroke-width="3" d="M.666.356v10.687"/></g><path id="text5474" fill="#181717" fill-rule="nonzero" d="M71.803 52.199l-1.188-1.172 1.264-1.326 1.219 1.203c.162-.309.286-.66.373-1.056a6.08 6.08 0 00.13-1.296c0-1.248-.278-2.217-.83-2.906-.554-.694-1.329-1.04-2.324-1.04-.985 0-1.751.344-2.3 1.032-.548.689-.822 1.66-.822 2.914 0 1.25.274 2.22.822 2.915.549.688 1.315 1.033 2.3 1.033.254 0 .493-.026.716-.077a2.86 2.86 0 00.64-.224m1.622 1.55c-.38.257-.825.452-1.333.585a6.39 6.39 0 01-1.645.2c-1.675 0-3.008-.539-3.998-1.618-.985-1.08-1.477-2.534-1.477-4.364 0-1.834.492-3.289 1.477-4.363.99-1.08 2.323-1.62 3.998-1.62 1.676 0 3.008.54 3.999 1.62.995 1.08 1.492 2.534 1.492 4.363 0 .802-.101 1.537-.304 2.205a5.234 5.234 0 01-.9 1.743l1.22 1.18-1.25 1.325-1.279-1.256"/><path id="text5482" fill="#181717" fill-rule="nonzero" d="M105.05 55.005l-3.14-11.399h2.337l1.876 8.04 1.585-8.04h2.404l1.585 8.04 1.875-8.04h2.315l-3.133 11.4h-2.151l-1.697-8.73-1.704 8.73h-2.151"/><path id="text5492" fill="#181717" fill-rule="nonzero" d="M144.839 55.005V43.606h8.692v1.981h-6.261v2.43h5.724v1.95h-5.724V52.9h6.554v2.105h-8.985"/><path id="text5500" fill="#181717" fill-rule="nonzero" d="M187.015 48.59h2.634c.573 0 .992-.121 1.259-.364.271-.242.407-.624.407-1.145 0-.495-.131-.87-.393-1.122-.261-.258-.65-.387-1.169-.387h-2.738v3.018m-2.243 6.415V43.606h5.284c1.19 0 2.07.266 2.643.797.572.532.858 1.344.858 2.438 0 .696-.14 1.277-.422 1.741a2.108 2.108 0 01-1.199.952c.474.175.81.444 1.007.805.202.36.318.923.348 1.687l.044 1.346v.047c.015.68.156 1.093.422 1.238v.348H191.3a2.318 2.318 0 01-.185-.588 6.4 6.4 0 01-.074-.843l-.03-1.2c-.024-.707-.153-1.184-.385-1.432-.227-.247-.639-.371-1.236-.371h-2.375v4.434h-2.243"/><path id="text5504" fill="#181717" fill-rule="nonzero" d="M228.05 55.005v-9.364h-3.345v-2.035h8.986v2.035h-3.322v9.364h-2.319"/><path id="text5518" fill="#181717" fill-rule="nonzero" d="M101.245 134.798H98.96v-4.249l-4.037-7.15h2.8l2.376 4.914 2.196-4.914h2.611l-3.66 7.15v4.249"/><path id="text5522" fill="#181717" fill-rule="nonzero" d="M304.572 43.606h2.33v6.939c0 .856.173 1.482.518 1.877.345.39.894.586 1.645.586.761 0 1.315-.196 1.66-.586.35-.39.525-1.016.525-1.877v-6.939h2.307v7.187c0 1.356-.385 2.397-1.157 3.123-.767.726-1.873 1.09-3.32 1.09-1.457 0-2.574-.361-3.35-1.082-.772-.726-1.158-1.77-1.158-3.131v-7.187"/><path id="text5526" fill="#181717" fill-rule="nonzero" d="M347.5 55.005V43.606h1.998v11.4H347.5"/><path id="text5530" fill="#181717" fill-rule="nonzero" d="M384.8 48.788c0 1.298.274 2.307.823 3.028.549.716 1.316 1.074 2.303 1.074.996 0 1.772-.358 2.326-1.074.554-.72.831-1.73.831-3.028s-.277-2.305-.831-3.021c-.554-.721-1.33-1.082-2.326-1.082-.987 0-1.754.358-2.303 1.074-.55.716-.824 1.725-.824 3.029m-2.356 0c0-1.907.493-3.419 1.48-4.535.99-1.122 2.325-1.683 4.003-1.683s3.012.56 4.004 1.683c.996 1.121 1.494 2.633 1.494 4.535 0 1.901-.498 3.413-1.494 4.535-.992 1.121-2.326 1.682-4.004 1.682-1.678 0-3.013-.56-4.004-1.682-.986-1.122-1.48-2.634-1.48-4.535"/><path id="text5534" fill="#181717" fill-rule="nonzero" d="M425.808 48.884h2.338c.633 0 1.093-.129 1.38-.387.288-.263.432-.684.432-1.261 0-.542-.141-.952-.423-1.23-.282-.28-.703-.419-1.261-.419h-2.466v3.297m-.016 2.066v4.055h-2.418V43.606h5.227c1.234 0 2.168.312 2.8.937.64.619.958 1.53.958 2.731 0 1.177-.31 2.085-.933 2.724-.623.635-1.511.952-2.665.952h-2.969"/><path id="path3102" fill="#C9DCEA" fill-rule="nonzero" d="M311.56 21.86v1.6c-.049 0-.118-.003-.206-.008a3.788 3.788 0 00-.182-.007c-.556 0-.933.1-1.131.298-.193.194-.29.608-.29 1.24v1.838c0 .755-.124 1.291-.371 1.608-.248.316-.694.543-1.339.68.645.139 1.09.363 1.339.675.247.311.371.844.371 1.6v1.845c0 .628.097 1.038.29 1.232.192.194.57.291 1.131.291.033 0 .094-.002.182-.007s.157-.008.207-.008v1.6c-.077 0-.185.003-.322.008-.138.005-.24.007-.306.007-.551 0-1.01-.03-1.38-.092a3.388 3.388 0 01-.942-.283 1.495 1.495 0 01-.67-.704c-.126-.291-.19-.799-.19-1.524V31.82c0-.7-.14-1.194-.42-1.485-.281-.296-.755-.444-1.422-.444-.033 0-.088.002-.165.007a2.87 2.87 0 01-.173.008v-1.6c.038 0 .096.003.173.008.077.005.132.007.165.007.661 0 1.132-.148 1.413-.444.287-.296.43-.796.43-1.5v-1.914c0-.73.063-1.243.19-1.539.132-.296.355-.53.67-.704.258-.128.572-.222.941-.284a8.644 8.644 0 011.686-.084c.137.005.245.008.322.008"/><path id="text5546" fill="#C9DCEA" fill-rule="nonzero" d="M347.5 22.88h3.994v1.62h-1.858v11.27h1.858v1.619h-3.993V22.88"/><path id="path3106" fill="#C9DCEA" fill-rule="nonzero" d="M425.371 21.845h.562c.606 0 1.093.03 1.462.092.37.06.678.158.926.29.314.164.534.4.66.706.133.301.199.807.199 1.517v1.932c0 .7.14 1.198.421 1.494.281.291.755.437 1.421.437.033 0 .088-.003.166-.008a2.88 2.88 0 01.173-.007v1.601h-.306c-.688 0-1.173.143-1.454.43-.28.286-.421.781-.421 1.486v1.932c0 .73-.066 1.244-.198 1.54a1.381 1.381 0 01-.661.69 3.386 3.386 0 01-.942.283 8.634 8.634 0 01-1.686.085 9.241 9.241 0 00-.322-.008v-1.602c.05 0 .118.003.207.008.088.005.148.008.181.008.557 0 .931-.1 1.124-.3.198-.193.297-.602.297-1.226v-1.831c0-.767.124-1.306.372-1.617.248-.317.694-.542 1.339-.675-.645-.138-1.09-.365-1.339-.682-.248-.317-.372-.853-.372-1.61v-1.854c0-.623-.099-1.032-.297-1.226-.193-.2-.567-.299-1.124-.299-.033 0-.093.003-.181.008a3.778 3.778 0 01-.207.007v-1.601"/><path id="text5554" fill="#C9DCEA" fill-rule="nonzero" d="M390.43 22.88V37.39h-3.994v-1.62h1.862V24.5h-1.862v-1.618h3.993"/><path id="text5641" fill="#181717" fill-rule="nonzero" d="M79.428 86.304l-1.594 4.852h3.196l-1.602-4.852m-1.376-2.283h2.736l4.15 11.399h-2.485l-.782-2.345h-4.463l-.758 2.345h-2.493l4.095-11.4"/><path id="text5645" fill="#181717" fill-rule="nonzero" d="M114.889 92.487h2.242c.085.645.333 1.126.743 1.44.41.31 1.002.464 1.777.464.66 0 1.158-.125 1.493-.376.335-.25.502-.621.502-1.112 0-.715-.962-1.307-2.887-1.776a6.848 6.848 0 01-.3-.072c-1.03-.24-1.765-.51-2.205-.809a2.56 2.56 0 01-.893-1.072c-.205-.453-.307-.987-.307-1.6 0-1.147.365-2.025 1.095-2.633.73-.614 1.785-.92 3.165-.92 1.29 0 2.297.325 3.022.976.73.65 1.115 1.568 1.155 2.753h-2.182c-.04-.571-.245-1.006-.615-1.305-.37-.298-.898-.448-1.583-.448-.595 0-1.055.126-1.38.376-.32.246-.48.598-.48 1.057 0 .624.628 1.09 1.883 1.4l.795.2c.805.219 1.375.39 1.71.512.34.123.635.259.885.408.45.267.787.622 1.012 1.065.225.437.338.962.338 1.576 0 1.227-.388 2.18-1.163 2.857-.775.672-1.867 1.008-3.277 1.008-1.39 0-2.48-.344-3.27-1.032-.79-.688-1.215-1.667-1.275-2.937"/><path id="text5649" fill="#181717" fill-rule="nonzero" d="M156.959 93.361h1.748c1.012 0 1.743-.284 2.194-.851.455-.573.683-1.499.683-2.778 0-1.274-.21-2.21-.633-2.81-.422-.598-1.081-.897-1.978-.897h-2.014v7.336m-2.137 2.059v-11.4h4.15c1.627 0 2.838.473 3.634 1.417.8.944 1.201 2.376 1.201 4.295 0 1.042-.149 1.96-.446 2.755-.293.794-.72 1.434-1.28 1.919a3.9 3.9 0 01-1.44.782c-.536.154-1.29.232-2.258.232h-3.561"/><path id="text5653" fill="#181717" fill-rule="nonzero" d="M194.755 95.42v-11.4h7.987v1.982h-5.664v2.584h4.965v1.982h-4.965v4.852h-2.323"/><path id="text5657" fill="#181717" fill-rule="nonzero" d="M241.917 94.733c-.4.593-.856 1.029-1.368 1.306-.508.278-1.106.417-1.794.417-1.507 0-2.73-.572-3.667-1.715-.931-1.148-1.397-2.654-1.397-4.519 0-1.88.466-3.384 1.397-4.51.932-1.128 2.174-1.691 3.726-1.691 1.352 0 2.454.355 3.308 1.065.854.705 1.362 1.678 1.523 2.917h-2.262c-.137-.614-.417-1.082-.842-1.402-.42-.32-.968-.481-1.647-.481-.902 0-1.612.363-2.13 1.09-.512.72-.768 1.72-.768 2.996 0 1.282.268 2.286.805 3.013.537.726 1.274 1.09 2.21 1.09.703 0 1.291-.227 1.764-.681.474-.454.767-1.077.879-1.867h-2.445v-2.02h4.465v6.386h-1.486l-.27-1.394"/><path id="text5661" fill="#181717" fill-rule="nonzero" d="M274.622 95.42v-11.4h2.25v4.25h4.485v-4.25h2.25v11.4h-2.25v-5.054h-4.485v5.054h-2.25"/><path id="text5665" fill="#181717" fill-rule="nonzero" d="M314.556 90.978h2.484v1.34c0 .386.13.682.39.888.26.2.633.301 1.12.301.53 0 .9-.123 1.11-.369.209-.246.313-.707.313-1.385v-7.732h2.57v7.837c0 .703-.049 1.22-.145 1.551a2.23 2.23 0 01-.458.866c-.327.372-.777.655-1.348.851-.57.196-1.24.294-2.009.294-.707 0-1.331-.086-1.874-.256-.542-.171-.995-.424-1.356-.76a2.392 2.392 0 01-.62-.904c-.118-.337-.177-.889-.177-1.657v-.865"/><path id="text5669" fill="#181717" fill-rule="nonzero" d="M353.49 95.42v-11.4h2.31v4.683l4.382-4.682h2.888l-4.48 4.612 4.884 6.787h-2.82l-3.658-5.216-1.196 1.207v4.009h-2.31"/><path id="text5673" fill="#181717" fill-rule="nonzero" d="M394.423 95.42v-11.4h2.368v9.295h5.618v2.105h-7.986"/><path id="text5679" fill="#181717" fill-rule="nonzero" d="M264.639 55.005v-1.98l6.156-7.353h-6.033v-2.066h8.862v1.981l-6.172 7.36h6.034v2.058h-8.847"/><path id="text5683" fill="#181717" fill-rule="nonzero" d="M134.855 134.798l3.55-5.82-3.55-5.579h2.684l2.308 3.985 2.293-3.985h2.699l-3.55 5.564 3.55 5.835h-2.684l-2.308-3.947-2.308 3.947h-2.684"/><path id="text5687" fill="#181717" fill-rule="nonzero" d="M184.772 131.451c-.084 1.341-.564 2.407-1.441 3.197-.872.79-2.01 1.186-3.417 1.186-1.62 0-2.88-.545-3.78-1.634-.897-1.09-1.345-2.618-1.345-4.583 0-2.009.458-3.547 1.374-4.616.916-1.068 2.233-1.602 3.952-1.602 1.396 0 2.498.369 3.305 1.106.812.731 1.258 1.768 1.337 3.108h-2.22c-.095-.667-.347-1.175-.758-1.522-.412-.352-.966-.529-1.664-.529-.99 0-1.743.345-2.259 1.034-.515.689-.772 1.696-.772 3.02 0 1.283.255 2.273.765 2.973.515.7 1.246 1.05 2.191 1.05.684 0 1.243-.187 1.68-.561.435-.38.717-.921.846-1.627h2.206"/><path id="text5691" fill="#181717" fill-rule="nonzero" d="M217.625 134.798l-3.901-11.399h2.58l2.404 8.528 2.45-8.528h2.55l-3.856 11.399h-2.227"/><path id="text5695" fill="#181717" fill-rule="nonzero" d="M256.813 132.74h2.67c.63 0 1.088-.12 1.376-.357.287-.237.431-.61.431-1.122 0-.531-.141-.913-.424-1.145-.282-.237-.749-.356-1.398-.356h-2.655v2.98m0-4.93h2.573c.555 0 .962-.098 1.22-.294.258-.201.387-.516.387-.944 0-.418-.127-.72-.38-.906-.248-.19-.662-.286-1.242-.286h-2.558v2.43m-2.157 6.988v-11.399h5.161c1.121 0 1.974.248 2.56.743.59.49.884 1.205.884 2.143 0 .578-.109 1.06-.327 1.448a2.183 2.183 0 01-.982.897c.565.238.987.58 1.265 1.03.282.443.424.998.424 1.663 0 1.11-.348 1.966-1.042 2.57-.694.603-1.678.905-2.953.905h-4.99"/><path id="text5699" fill="#181717" fill-rule="nonzero" d="M294.589 134.798v-11.399h2.346l4.426 7.831V123.4h2.213v11.399h-2.317l-4.455-7.832v7.832h-2.213"/><path id="text5703" fill="#181717" fill-rule="nonzero" d="M333.524 134.798v-11.399h3.324l2.174 8.675 2.145-8.675h3.339v11.399h-2.107v-9.194l-2.22 9.194h-2.3l-2.248-9.194v9.194h-2.107"/><g id="g3476" transform="translate(561.145 70.55)"><path id="path5745" fill="#181717" d="M10.246 14.192V4.731L0 9.46z"/><path id="path5747" stroke="#181717" stroke-width="3" d="M25.615 0v9.462H5.123"/></g><path id="path5757" stroke="#181717" stroke-width="2" d="M18.55 130.653l-7.488 9.672h4.991v4.836h4.992v-4.836h4.992z"/><path id="path5801" stroke="#181717" stroke-width="2" d="M501.744 130.653l-7.488 9.672h4.992v4.836h4.992v-4.836h4.991z"/><path id="path3083" fill="#181717" fill-rule="nonzero" d="M447.271 12.398c0-.055-.002-.128-.008-.219a4.189 4.189 0 01-.008-.21c0-.443.051-.81.153-1.102.101-.297.264-.574.488-.83.171-.191.425-.41.761-.656.342-.247.564-.425.665-.536.214-.226.36-.43.44-.611.08-.181.12-.38.12-.596 0-.478-.14-.85-.424-1.117-.283-.266-.678-.4-1.185-.4-.508 0-.908.161-1.202.483-.288.317-.446.765-.473 1.343h-2.259v-.234c0-1.06.356-1.906 1.066-2.534.715-.634 1.677-.951 2.884-.951 1.233 0 2.213.3 2.94.898.731.593 1.097 1.39 1.097 2.391 0 .352-.043.67-.128.95-.08.277-.206.531-.377.763-.219.291-.558.608-1.017.95-.454.337-.729.546-.825.626-.203.186-.35.383-.44.589a1.927 1.927 0 00-.12.867c.005.06.007.106.007.136h-2.155m-.112 3.229v-2.21h2.363v2.21h-2.363"/><path id="text5819" fill="#181717" fill-rule="nonzero" d="M286.602 17.7l3.912-13.472h1.08l-3.927 13.471h-1.065"/><path id="text5928" fill="#C9DCEA" fill-rule="nonzero" d="M173.79 71.348c-.084 1.34-.564 2.406-1.44 3.197-.872.79-2.011 1.186-3.418 1.186-1.619 0-2.88-.545-3.78-1.635-.897-1.09-1.345-2.617-1.345-4.583 0-2.008.458-3.547 1.374-4.615.916-1.068 2.234-1.603 3.952-1.603 1.396 0 2.498.369 3.305 1.106.813.732 1.258 1.768 1.337 3.109h-2.22c-.095-.668-.347-1.175-.758-1.522-.411-.353-.966-.53-1.664-.53-.99 0-1.743.345-2.258 1.034-.515.69-.773 1.696-.773 3.021 0 1.282.255 2.273.765 2.973.515.7 1.246 1.05 2.192 1.05.683 0 1.243-.188 1.678-.562.436-.379.718-.921.847-1.626h2.206"/><path id="text5956" fill="#C9DCEA" fill-rule="nonzero" d="M161.288 70.602h7.481l-.475.983h-7.482l.476-.983m1.027-2.125h7.482l-.476.983h-7.481l.475-.983"/><path id="path3060" fill="#181717" fill-rule="nonzero" d="M246.307 31.467c0 .634.177 1.136.53 1.507.36.371.846.557 1.46.557.573 0 1.022-.18 1.347-.541.324-.36.486-.863.486-1.507 0-.593-.168-1.054-.504-1.383-.336-.33-.808-.495-1.416-.495-.597 0-1.063.165-1.4.495-.335.33-.503.785-.503 1.367m-.035-2.89c.284-.257.626-.45 1.025-.58.4-.133.846-.2 1.339-.2 1.216 0 2.19.33 2.92.989.735.66 1.103 1.538 1.103 2.635 0 1.164-.4 2.105-1.2 2.82-.799.717-1.853 1.075-3.162 1.075-1.512 0-2.662-.464-3.45-1.391-.782-.927-1.173-2.285-1.173-4.073 0-1.921.42-3.39 1.26-4.405.84-1.02 2.05-1.53 3.632-1.53 1.1 0 1.987.255 2.66.765.677.51 1.048 1.2 1.111 2.071h-2.52c-.058-.355-.214-.623-.469-.803-.255-.18-.611-.27-1.069-.27-.625 0-1.106.241-1.442.726-.336.479-.524 1.203-.565 2.171"/><path id="rect10188" fill="#FFF" stroke="#181717" d="M508.233 79.876h39.933v39.378h-39.933z"/><path id="path2917" fill="#C9DCEA" fill-rule="nonzero" d="M532.193 69.513v1.784c-.512.383-.997.665-1.455.848a3.66 3.66 0 01-1.344.267c-.226 0-.452-.022-.679-.065a4.552 4.552 0 01-.68-.186 22.53 22.53 0 01-.524-.201c-1.078-.404-1.897-.606-2.458-.606-.419 0-.857.102-1.315.307-.453.199-.962.52-1.528.96v-1.784c.526-.387 1.029-.675 1.506-.864.478-.194.928-.29 1.351-.29.576 0 1.293.156 2.15.468a.111.111 0 00.029.008c.088.032.224.083.406.153.763.296 1.354.444 1.772.444.418 0 .85-.1 1.292-.298.443-.2.935-.514 1.477-.945"/><path id="path2915" fill="#181717" fill-rule="nonzero" d="M517.137 108.474l-.651 1.887h1.856l.636-1.887h-1.841m0-4.764h1.634l-1.08 3.21h1.79l1.094-3.21h1.635l-1.095 3.21h2.093l-.547 1.554h-2.07l-.622 1.871h2.145l-.526 1.547h-2.152l-1.094 3.217h-1.634l1.094-3.217h-1.804l-1.102 3.217h-1.627l1.08-3.217h-2.123l.562-1.547h2.086l.636-1.871h-2.211l.547-1.555h2.196l1.095-3.21"/><path id="rect10196" fill="#FFF" stroke="#181717" d="M49.997 119.254H89.93v39.378H49.997z"/><path id="text5910" fill="#C9DCEA" fill-rule="nonzero" d="M106.902 31.171c.018-.655.181-1.209.49-1.661.316-.452.87-.91 1.661-1.373l.473-.262c1.101-.594 1.652-1.167 1.652-1.72 0-.323-.11-.577-.33-.763-.22-.187-.524-.28-.911-.28-.428 0-.759.116-.99.347-.233.226-.349.551-.349.975v.068h-1.553c0-.83.259-1.47.776-1.915.518-.447 1.259-.67 2.223-.67.869 0 1.56.203 2.071.61.518.401.777.938.777 1.61 0 .486-.134.904-.402 1.255-.268.35-.791.754-1.571 1.211-.173.108-.417.249-.732.424-.714.395-1.116.706-1.205.932h3.838v1.212h-5.918"/><path id="text5922" fill="#C9DCEA" fill-rule="nonzero" d="M146.835 29.844h1.603c.012.4.127.7.344.898.217.197.543.296.978.296.417 0 .74-.093.969-.28.229-.187.343-.453.343-.799 0-.35-.132-.614-.396-.79-.264-.181-.66-.272-1.19-.272h-.255v-.98h.185c.488 0 .846-.076 1.075-.23.235-.16.352-.4.352-.725 0-.274-.1-.49-.3-.65-.199-.165-.466-.247-.8-.247-.37 0-.656.093-.855.28-.2.181-.3.442-.3.782v.05h-1.568c.018-.72.265-1.269.74-1.647.482-.385 1.166-.577 2.053-.577.834 0 1.491.17 1.973.51.487.341.731.805.731 1.392 0 .308-.085.58-.255.815-.165.231-.412.42-.74.569.428.159.751.381.969.667.223.28.334.609.334.988 0 .713-.276 1.279-.828 1.696-.546.411-1.292.617-2.237.617-.934 0-1.656-.203-2.167-.609-.505-.406-.758-.98-.758-1.72v-.034"/><path id="path3131" fill="#181717" fill-rule="nonzero" d="M131.142 12.276c.232-.12.406-.254.522-.4a.747.747 0 00.174-.472.728.728 0 00-.2-.519c-.133-.147-.409-.317-.826-.51l-2.306-1.078c-.226.121-.4.257-.522.41a.789.789 0 00-.174.486c0 .205.084.386.253.543.174.152.554.364 1.14.636l1.94.904m-4.733 2.57h2.149c.046.41.18.713.4.912.226.2.545.299.957.299.383 0 .687-.081.913-.244a.799.799 0 00.34-.676c0-.23-.085-.43-.253-.597-.168-.173-.501-.37-1-.59l-2.08-.95c-.66-.3-1.154-.642-1.479-1.03a1.98 1.98 0 01-.487-1.305c0-.466.125-.875.374-1.226.256-.356.63-.652 1.123-.888a2.377 2.377 0 01-.644-.77 2.09 2.09 0 01-.209-.936c0-.786.299-1.417.896-1.894.598-.482 1.392-.723 2.384-.723 1.02 0 1.821.236 2.401.707.58.472.879 1.124.896 1.957h-2.053c-.012-.35-.128-.618-.348-.801-.215-.184-.525-.275-.93-.275-.36 0-.639.07-.836.212a.7.7 0 00-.287.597c0 .21.113.406.34.59.231.178.646.393 1.243.644l1.462.613c.742.32 1.29.676 1.644 1.07.354.387.53.835.53 1.343 0 .477-.133.894-.4 1.25-.26.35-.637.623-1.13.817.336.267.588.564.757.888.168.325.252.682.252 1.07 0 .806-.32 1.472-.957 1.996-.638.529-1.459.793-2.462.793-1.102 0-1.95-.24-2.54-.723-.592-.482-.914-1.192-.966-2.13"/><path id="text6073" fill="#181717" fill-rule="nonzero" d="M444.34 35.039v-8.34c0-.83.334-1.5 1.002-2.01.668-.514 1.559-.772 2.672-.772 1.135 0 2.023.25 2.665.75.642.5.963 1.187.963 2.062 0 .54-.095.983-.287 1.328-.186.345-.477.61-.87.795.611.235 1.07.58 1.375 1.035.31.455.466 1.014.466 1.68 0 1.12-.334 2.024-1.002 2.714-.668.69-1.544 1.035-2.626 1.035-.15 0-.319-.007-.505-.022a10.34 10.34 0 01-.59-.068v-1.755a2.769 2.769 0 00.52.03c.611 0 1.083-.162 1.414-.487.331-.33.497-.798.497-1.403 0-.585-.181-1.03-.544-1.334-.357-.305-.88-.458-1.569-.458h-.21l.008-1.537.101.015h.163c.461 0 .81-.11 1.049-.33.243-.225.365-.55.365-.975 0-.425-.127-.748-.38-.968-.25-.225-.614-.337-1.096-.337-.445 0-.785.125-1.018.375-.233.245-.35.605-.35 1.08v7.897h-2.214"/><path id="path2750" fill="#181717" fill-rule="nonzero" d="M417.384 129.151v-2.643h1.997v2.643h-1.997m0 6.683v-2.626h1.997v2.626h-1.997"/><path id="path2746" fill="#181717" fill-rule="nonzero" d="M377.464 128.949v-2.441h1.984v2.44h-1.984m0 8.959v-.943c.37-.129.641-.327.817-.596.176-.268.264-.618.264-1.047v-.202h-1.094v-2.424h1.997v2.328c0 .805-.167 1.45-.501 1.933-.334.489-.828.806-1.483.95"/><path id="text5803" fill="#181717" fill-rule="nonzero" d="M377.45 157.596v-.937c.37-.128.642-.326.818-.593.176-.267.264-.614.264-1.04v-.201h-1.081v-2.41h1.997v2.314c0 .795-.17 1.436-.508 1.922-.334.486-.83.8-1.49.945"/><path id="text5813" fill="#181717" fill-rule="nonzero" d="M417.384 154.487v-2.072h1.997v2.072h-1.997"/><path id="path2737" fill="#181717" fill-rule="nonzero" d="M63.973 145.16v1.75l-6.81 2.918 6.81 2.909v1.75l-8.985-3.876v-1.59l8.985-3.86"/><path id="path2742" fill="#181717" fill-rule="nonzero" d="M54.988 125.472l8.985 3.86v1.59l-8.985 3.876v-1.75l6.827-2.91-6.827-2.916v-1.75"/><path id="text6003" fill="#181717" fill-rule="nonzero" d="M477.764 86.458l-1.594 4.77h3.196l-1.602-4.77m-1.375-2.244h2.735l4.15 11.206h-2.485l-.782-2.305h-4.463l-.758 2.305h-2.493l4.096-11.206m1.868-1.43v-1.872h1.766v1.871h-1.766m-2.72 0v-1.871h1.766v1.871h-1.766"/><path id="text6023" fill="#181717" fill-rule="nonzero" d="M464.306 43.7h2.33v6.881c0 .85.173 1.47.518 1.862.345.388.893.581 1.645.581.761 0 1.314-.193 1.66-.58.35-.388.525-1.009.525-1.863V43.7h2.307v7.127c0 1.346-.386 2.378-1.157 3.098-.767.72-1.873 1.08-3.32 1.08-1.457 0-2.574-.357-3.35-1.072-.772-.72-1.158-1.755-1.158-3.106V43.7m4.972-1.37v-1.833H471v1.833h-1.72m-2.65 0v-1.833h1.72v1.833h-1.72"/><path id="text6041" fill="#181717" fill-rule="nonzero" d="M434.716 89.64c0 1.207.274 2.145.824 2.816.549.665 1.316.998 2.303.998.996 0 1.771-.333 2.326-.998.554-.67.83-1.61.83-2.816 0-1.206-.276-2.142-.83-2.807-.555-.67-1.33-1.006-2.326-1.006-.987 0-1.754.333-2.303.998-.55.666-.824 1.604-.824 2.815m-2.357 0c0-1.772.494-3.177 1.48-4.215.991-1.043 2.326-1.564 4.004-1.564 1.677 0 3.012.521 4.003 1.564.997 1.043 1.495 2.448 1.495 4.215 0 1.768-.498 3.173-1.495 4.216-.991 1.042-2.326 1.564-4.003 1.564-1.678 0-3.013-.522-4.004-1.564-.986-1.043-1.48-2.448-1.48-4.216m5.964-6.896v-1.832h1.724v1.832h-1.724m-2.654 0v-1.832h1.724v1.832h-1.724"/><path id="path2893" fill="#C9DCEA" fill-rule="nonzero" d="M90.991 73.43c-.274.422-.59.733-.949.934-.359.195-.782.293-1.27.293-.788 0-1.388-.215-1.802-.647-.408-.431-.613-1.069-.613-1.912 0-1.199.347-2.205 1.04-3.018.697-.812 1.554-1.219 2.57-1.219.4 0 .745.083 1.04.249.293.165.535.411.724.737l.397-.737h1.33l-1.099 4.5a1.757 1.757 0 00-.052.309c0 .175.06.308.18.399.119.085.296.128.53.128.194 0 .396-.05.605-.151.215-.1.414-.24.598-.421a3.96 3.96 0 00.972-1.408 4.545 4.545 0 00.336-1.768c0-1.205-.463-2.193-1.39-2.966-.922-.777-2.113-1.166-3.573-1.166a7.25 7.25 0 00-1.965.256 5.609 5.609 0 00-1.645.76 5.949 5.949 0 00-1.958 2.145 5.946 5.946 0 00-.68 2.822c0 1.54.516 2.772 1.547 3.695 1.032.918 2.41 1.378 4.133 1.378.808 0 1.595-.123 2.362-.37a7.857 7.857 0 002.153-1.068l.613.88a8.585 8.585 0 01-2.43 1.303 8.199 8.199 0 01-2.668.436c-.956 0-1.836-.123-2.638-.369a6.607 6.607 0 01-2.138-1.076c-.772-.607-1.35-1.307-1.734-2.1-.383-.792-.575-1.69-.575-2.694 0-.838.13-1.636.388-2.393a6.747 6.747 0 011.159-2.077 7.198 7.198 0 012.66-2.062c1.037-.467 2.19-.7 3.461-.7.842 0 1.632.115 2.37.346a6.101 6.101 0 011.935.963c.673.527 1.174 1.129 1.503 1.806.334.673.5 1.433.5 2.28 0 .829-.164 1.594-.493 2.296a4.984 4.984 0 01-1.405 1.791c-.349.281-.73.497-1.144.648a3.781 3.781 0 01-1.278.218c-.528 0-.926-.106-1.195-.316-.265-.211-.402-.522-.412-.933m.374-3.222c-.065-.406-.21-.715-.433-.925-.22-.216-.511-.324-.875-.324-.568 0-1.061.311-1.48.933-.418.622-.628 1.37-.628 2.243 0 .462.1.813.3 1.054.204.24.498.36.881.36.414 0 .79-.157 1.129-.473.344-.316.578-.728.703-1.234l.403-1.634"/><path id="path2898" fill="#C9DCEA" fill-rule="nonzero" d="M469.216 37.389l-3.912-13.472h1.065l3.927 13.472h-1.08"/><path id="path2919" fill="#181717" fill-rule="nonzero" d="M516.22 82.984h1.996v5.182h-1.996v-5.182"/><path id="text6099" fill="#D35155" fill-rule="nonzero" d="M490.263 22.88l-2.747 3.11h-1.246l1.729-3.11h2.264"/><path id="text5103" fill="#D35155" fill-rule="nonzero" d="M485.271 3.192h2.265l1.729 3.109h-1.247l-2.747-3.11"/><path id="text6117" fill="#C9DCEA" fill-rule="nonzero" d="M356.42 146.197h2.083l-.548 3.24c-.072.398-.127.776-.166 1.133a8.3 8.3 0 00-.057.864c0 .462.1.808.302 1.036.202.224.51.335.922.335.48 0 .853-.203 1.117-.61.27-.408.493-1.13.67-2.169l.649-3.83h2.082l-1.39 8.137h-1.931l.13-.872c-.202.387-.428.67-.678.85a1.439 1.439 0 01-.872.267c-.322 0-.608-.074-.857-.223-.25-.144-.476-.37-.678-.678l-.648 3.919h-2.06l1.93-11.4"/><path id="path3396" fill="#C9DCEA" fill-rule="nonzero" d="M78.948 141.016h1.997v15.544h-1.997v-15.544"/><path id="path3403" fill="#181717" fill-rule="nonzero" d="M9.564 5.327c-.4 0-.744.145-1.03.435-.28.29-.42.64-.42 1.048 0 .42.14.776.42 1.065.28.29.624.435 1.03.435.395 0 .733-.148 1.013-.443.28-.296.42-.648.42-1.057 0-.409-.14-.758-.42-1.048a1.358 1.358 0 00-1.013-.435m0-1.099c.692 0 1.28.256 1.763.767a2.54 2.54 0 01.733 1.832 2.53 2.53 0 01-.733 1.824 2.382 2.382 0 01-1.78.758 2.36 2.36 0 01-1.762-.75c-.478-.5-.717-1.11-.717-1.832 0-.721.242-1.335.725-1.84a2.373 2.373 0 011.771-.759"/><path id="text6091" fill="#D35155" fill-rule="nonzero" d="M8.632 23.917h1.865l3.56 5.181h-1.74l-2.746-3.426-2.76 3.426H5.07l3.56-5.18"/><path id="path2585" stroke="#181717" stroke-width="2" d="M18.55 109.927l7.487-9.671h-4.992V95.42h-4.992v4.836h-4.991z"/><text id="Strg" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="10.982" y="185.293">Strg</tspan></text><text id="tspan3451" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="553.078" y="185.293">Strg</tspan></text><g id="text3456" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" transform="translate(116.805 167.948)"><text id="tspan3460"><tspan x="0" y="17.345">Al</tspan> <tspan x="0" y="39.345">t</tspan></text></g><text id="tspan3466" fill="#C9DCEA" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="405.324" y="185.293">Alt Gr</tspan></text><text id="Win" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="64.892" y="185.293">Win</tspan></text><text id="Win" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="464.226" y="185.293">Win</tspan></text><text id="Menu" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="499.171" y="185.293">Menu</tspan></text><circle id="Oval" cx="109.94" cy="138" r="26" stroke="#C06334" stroke-width="5"/></g></g></svg> \ No newline at end of file diff --git a/2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/index.html b/2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/index.html similarity index 95% rename from 2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/index.html rename to 2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/index.html index 401062830..a0d5a4f40 100644 --- a/2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/index.html +++ b/2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/index.html @@ -28,7 +28,7 @@ <input type="text" placeholder="Press keys here" id="kinput"> - <textarea id="area"></textarea> + <textarea id="area" readonly></textarea> <input type="button" value="Clear" onclick="area.value = ''" /> </form> <script src="script.js"></script> diff --git a/2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/script.js b/2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/script.js similarity index 96% rename from 2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/script.js rename to 2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/script.js index 5eba24c7a..d97f7a7b5 100644 --- a/2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/script.js +++ b/2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/script.js @@ -5,6 +5,8 @@ let lastTime = Date.now(); function handle(e) { if (form.elements[e.type + 'Ignore'].checked) return; + area.scrollTop = 1e6; + let text = e.type + ' key=' + e.key + ' code=' + e.code + diff --git a/2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/style.css b/2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/style.css similarity index 100% rename from 2-ui/3-event-details/5-keyboard-events/keyboard-dump.view/style.css rename to 2-ui/3-event-details/7-keyboard-events/keyboard-dump.view/style.css diff --git a/2-ui/3-event-details/7-keyboard-events/us-layout.svg b/2-ui/3-event-details/7-keyboard-events/us-layout.svg new file mode 100644 index 000000000..353f225f1 --- /dev/null +++ b/2-ui/3-event-details/7-keyboard-events/us-layout.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="602" height="202" viewBox="0 0 602 202"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="KB_United_States-NoAltGr" transform="translate(1 1)"><path id="rect2186" fill="#FFF" stroke="#181717" d="M0 0h39.604v40H0z"/><path id="rect2218" fill="#FFF" stroke="#181717" d="M40.594 0h39.604v40H40.594z"/><path id="rect2222" fill="#FFF" stroke="#181717" d="M80.198 0h39.604v40H80.198z"/><path id="rect2228" fill="#FFF" stroke="#181717" d="M119.802 0h39.604v40h-39.604z"/><path id="rect2230" fill="#FFF" stroke="#181717" d="M160.396 0H200v40h-39.604z"/><path id="rect2232" fill="#FFF" stroke="#181717" d="M200 0h39.604v40H200z"/><path id="rect2234" fill="#FFF" stroke="#181717" d="M240.594 0h39.604v40h-39.604z"/><path id="rect2236" fill="#FFF" stroke="#181717" d="M280.198 0h39.604v40h-39.604z"/><path id="rect2238" fill="#FFF" stroke="#181717" d="M319.802 0h39.604v40h-39.604z"/><path id="rect2240" fill="#FFF" stroke="#181717" d="M360.396 0H400v40h-39.604z"/><path id="rect2242" fill="#FFF" stroke="#181717" d="M400 0h39.604v40H400z"/><path id="rect2244" fill="#FFF" stroke="#181717" d="M440.594 0h39.604v40h-39.604z"/><path id="rect2246" fill="#FFF" stroke="#181717" d="M480.198 0h39.604v40h-39.604z"/><path id="rect2248" fill="#D1CFCD" stroke="#181717" d="M519.802 0H600v40h-80.198z"/><path id="rect2250" fill="#D1CFCD" stroke="#181717" d="M0 40h60.396v40H0z"/><path id="rect2252" fill="#FFF" stroke="#181717" d="M60.396 40H100v40H60.396z"/><path id="rect2254" fill="#FFF" stroke="#181717" d="M100 40h39.604v40H100z"/><path id="rect2256" fill="#FFF" stroke="#181717" d="M140.594 40h39.604v40h-39.604z"/><path id="rect2258" fill="#FFF" stroke="#181717" d="M180.198 40h39.604v40h-39.604z"/><path id="rect2262" fill="#FFF" stroke="#181717" d="M219.802 40h39.604v40h-39.604z"/><path id="rect2264" fill="#FFF" stroke="#181717" d="M260.396 40H300v40h-39.604z"/><path id="rect2266" fill="#FFF" stroke="#181717" d="M300 40h39.604v40H300z"/><path id="rect2270" fill="#FFF" stroke="#181717" d="M340.594 40h39.604v40h-39.604z"/><path id="rect2272" fill="#FFF" stroke="#181717" d="M380.198 40h39.604v40h-39.604z"/><path id="rect2274" fill="#FFF" stroke="#181717" d="M419.802 40h39.604v40h-39.604z"/><path id="rect2278" fill="#FFF" stroke="#181717" d="M460.396 40H500v40h-39.604z"/><path id="rect2280" fill="#FFF" stroke="#181717" d="M500 40h39.604v40H500z"/><path id="rect2284" fill="#FFF" stroke="#181717" d="M539.604 40H600v40h-60.396z"/><path id="rect2286" fill="#D1CFCD" stroke="#181717" d="M0 80h70.297v40H0z"/><path id="rect2292" fill="#FFF" stroke="#181717" d="M70.297 80h39.604v40H70.297z"/><path id="rect2296" fill="#FFF" stroke="#181717" d="M109.901 80h39.604v40h-39.604z"/><path id="rect2298" fill="#FFF" stroke="#181717" d="M150.495 80h39.604v40h-39.604z"/><path id="rect2300" fill="#FFF" stroke="#181717" d="M190.099 80h39.604v40h-39.604z"/><path id="rect2302" fill="#FFF" stroke="#181717" d="M230.693 80h39.604v40h-39.604z"/><path id="rect2306" fill="#FFF" stroke="#181717" d="M270.297 80h39.604v40h-39.604z"/><path id="rect2308" fill="#FFF" stroke="#181717" d="M309.901 80h39.604v40h-39.604z"/><path id="rect2312" fill="#FFF" stroke="#181717" d="M350.495 80h39.604v40h-39.604z"/><path id="rect2314" fill="#FFF" stroke="#181717" d="M390.099 80h39.604v40h-39.604z"/><path id="rect2316" fill="#FFF" stroke="#181717" d="M430.693 80h39.604v40h-39.604z"/><path id="rect2318" fill="#FFF" stroke="#181717" d="M470.297 80h39.604v40h-39.604z"/><path id="rect2320" fill="#D1CFCD" stroke="#181717" d="M509.901 80H600v40h-90.099z"/><path id="rect2322" fill="#D1CFCD" stroke="#181717" d="M0 120h90.099v40H0z"/><path id="rect2324" fill="#FFF" stroke="#181717" d="M90.099 120h39.604v40H90.099z"/><path id="rect2326" fill="#FFF" stroke="#181717" d="M130.693 120h39.604v40h-39.604z"/><path id="rect2330" fill="#FFF" stroke="#181717" d="M170.297 120h39.604v40h-39.604z"/><path id="rect2334" fill="#FFF" stroke="#181717" d="M209.901 120h39.604v40h-39.604z"/><path id="rect2336" fill="#FFF" stroke="#181717" d="M250.495 120h39.604v40h-39.604z"/><path id="rect2338" fill="#FFF" stroke="#181717" d="M290.099 120h39.604v40h-39.604z"/><path id="rect2340" fill="#FFF" stroke="#181717" d="M330.693 120h39.604v40h-39.604z"/><path id="rect2342" fill="#FFF" stroke="#181717" d="M370.297 120h39.604v40h-39.604z"/><path id="rect2344" fill="#FFF" stroke="#181717" d="M409.901 120h39.604v40h-39.604z"/><path id="rect2346" fill="#FFF" stroke="#181717" d="M450.495 120h39.604v40h-39.604z"/><path id="rect2348" fill="#D1CFCD" stroke="#181717" d="M490.099 120H600v40H490.099z"/><path id="rect2350" fill="#D1CFCD" stroke="#181717" d="M0 160h60.396v40H0z"/><path id="rect2352" fill="#D1CFCD" stroke="#181717" d="M539.604 160H600v40h-60.396z"/><path id="rect2354" fill="#D1CFCD" stroke="#181717" d="M100 160h60.396v40H100z"/><path id="rect2360" fill="#D1CFCD" stroke="#181717" d="M400 160h60.396v40H400z"/><path id="rect2362" fill="#FFF" stroke="#181717" d="M160.396 160H400v40H160.396z"/><path id="rect2364" fill="#F6F3F1" stroke="#181717" d="M60.396 160H100v40H60.396z"/><path id="rect2366" fill="#F6F3F1" stroke="#181717" d="M460.396 160H500v40h-39.604z"/><path id="rect2368" fill="#F6F3F1" stroke="#181717" d="M500 160h39.604v40H500z"/><path id="text5091" fill="#181717" fill-rule="nonzero" d="M50.238 12.434l-.619-5.26V4h2.444v3.174l-.62 5.26h-1.205m-.587 3.254v-2.341h2.356v2.34h-2.356M50.238 36v-7.768h-2.713v-1.595h.174c.99 0 1.709-.153 2.158-.46.455-.307.722-.82.801-1.54h1.817V36h-2.237"/><path id="text5103" fill="#181717" fill-rule="nonzero" d="M15.842 9v1.75c-.559.376-1.088.653-1.587.832a4.387 4.387 0 01-1.466.262c-.248 0-.495-.021-.742-.063a5.413 5.413 0 01-.74-.183c-.13-.042-.32-.108-.573-.198-1.176-.396-2.07-.594-2.682-.594-.457 0-.935.1-1.434.301-.494.196-1.05.51-1.668.943V10.3a7.651 7.651 0 011.644-.848 4.307 4.307 0 011.474-.286c.628 0 1.41.154 2.344.46a.133.133 0 00.032.008c.097.031.245.082.443.15.833.29 1.477.436 1.934.436.456 0 .926-.098 1.41-.293.483-.196 1.02-.505 1.61-.927M8.037 23.069h2.078L11.701 26h-1.144l-2.521-2.931"/><path id="text5127" fill="#181717" fill-rule="nonzero" d="M90.732 13.574c-.291.443-.628.77-1.009.982-.381.206-.831.309-1.35.309-.837 0-1.475-.227-1.914-.681-.434-.454-.651-1.125-.651-2.012 0-1.261.368-2.32 1.104-3.175.74-.855 1.652-1.283 2.732-1.283.423 0 .791.087 1.104.261.312.174.569.433.77.776l.42-.776h1.415l-1.168 4.736a1.833 1.833 0 00-.055.324c0 .185.063.325.19.42.127.09.315.135.564.135.206 0 .42-.053.643-.159.228-.105.44-.253.636-.443.455-.422.8-.916 1.032-1.48a4.741 4.741 0 00.358-1.862c0-1.267-.493-2.307-1.478-3.12-.98-.818-2.245-1.227-3.796-1.227-.747 0-1.443.09-2.089.269-.64.18-1.223.446-1.747.8-.9.602-1.594 1.354-2.08 2.257a6.207 6.207 0 00-.723 2.97c0 1.62.548 2.916 1.644 3.887 1.096.966 2.56 1.45 4.391 1.45a8.25 8.25 0 002.51-.389 8.382 8.382 0 002.287-1.124l.652.926a9.152 9.152 0 01-2.582 1.37 8.79 8.79 0 01-2.835.46c-1.016 0-1.95-.13-2.803-.388a7.047 7.047 0 01-2.272-1.133c-.82-.638-1.435-1.375-1.842-2.21-.408-.833-.612-1.778-.612-2.834 0-.882.138-1.721.413-2.518a7.08 7.08 0 011.231-2.186 7.64 7.64 0 012.828-2.17c1.1-.49 2.327-.736 3.677-.736.894 0 1.734.121 2.517.364.79.238 1.475.576 2.057 1.014.715.554 1.247 1.188 1.597 1.9.354.708.532 1.508.532 2.4 0 .87-.175 1.676-.524 2.415a5.252 5.252 0 01-1.494 1.885 4.054 4.054 0 01-2.573.91c-.561 0-.985-.11-1.27-.332-.281-.222-.427-.55-.437-.982m.397-3.39c-.069-.427-.223-.752-.46-.973-.234-.227-.544-.34-.93-.34-.604 0-1.128.326-1.573.981-.444.655-.667 1.441-.667 2.36 0 .486.106.855.318 1.108.217.254.53.38.937.38.44 0 .84-.166 1.2-.498.365-.333.614-.766.746-1.299l.429-1.718M85.37 36c.01-1.014.228-1.871.651-2.574.424-.707 1.176-1.422 2.256-2.146.164-.11.402-.264.714-.459 1.43-.913 2.145-1.792 2.145-2.637 0-.501-.151-.897-.453-1.188-.302-.29-.715-.435-1.239-.435-.572 0-1.014.18-1.326.538-.307.354-.46.86-.46 1.52v.096H85.56c0-1.299.35-2.3 1.049-3.002s1.694-1.053 2.986-1.053c1.17 0 2.105.32 2.804.958.698.634 1.048 1.479 1.048 2.534 0 .755-.183 1.41-.548 1.964-.365.555-1.072 1.185-2.12 1.893-.223.153-.53.354-.922.602-.995.639-1.557 1.124-1.684 1.457h5.163V36H85.37"/><path id="text5139" fill="#181717" fill-rule="nonzero" d="M130.065 8.85l-.704 1.92h2.008l.688-1.92h-1.992m0-4.85h1.768l-1.168 3.267h1.936L133.785 4h1.769l-1.185 3.267h2.265l-.592 1.582h-2.24l-.673 1.905h2.32l-.568 1.575h-2.328l-1.184 3.274h-1.768l1.184-3.274h-1.952l-1.192 3.274h-1.76l1.168-3.274h-2.297l.609-1.575h2.256l.688-1.905h-2.392l.592-1.582h2.376L130.065 4m-3.552 28.245h2.2c.005.63.16 1.105.464 1.425.304.315.755.472 1.352.472.576 0 1.019-.147 1.328-.44.315-.3.472-.727.472-1.284 0-.556-.184-.976-.552-1.26-.363-.288-.904-.432-1.624-.432-.037 0-.096.002-.176.008-.075.005-.13.007-.168.007v-1.558h.248c.661 0 1.15-.126 1.464-.378.32-.257.48-.648.48-1.173 0-.436-.136-.78-.408-1.031-.267-.257-.632-.386-1.096-.386-.507 0-.899.147-1.176.44-.277.295-.416.712-.416 1.253v.094h-2.16c.026-1.155.37-2.039 1.032-2.653.667-.62 1.605-.929 2.816-.929 1.147 0 2.056.276 2.728.827.672.55 1.008 1.293 1.008 2.228 0 .493-.114.926-.344 1.298-.229.373-.573.68-1.032.921.587.252 1.03.601 1.328 1.047.304.441.456.969.456 1.583 0 1.128-.376 2.023-1.128 2.684-.752.661-1.778.992-3.08.992-1.285 0-2.278-.32-2.976-.96-.694-.646-1.04-1.562-1.04-2.748v-.047"/><path id="text5145" fill="#181717" fill-rule="nonzero" d="M169.832 14.64c.412-.056.727-.21.944-.458.216-.25.324-.581.324-.996 0-.368-.103-.671-.31-.91-.2-.244-.52-.433-.958-.568v2.933m-.899-5.32V6.675c-.392.052-.692.192-.898.42-.207.223-.31.524-.31.902 0 .316.096.58.287.793.196.213.503.39.921.53m0 8.517v-1.494c-1.137-.072-1.988-.412-2.552-1.019-.564-.606-.846-1.483-.846-2.629h2.024c.025.56.15 1.004.377 1.33.232.322.564.527.997.615v-3.22c-1.168-.306-1.998-.706-2.492-1.198-.493-.493-.74-1.167-.74-2.023 0-.949.292-1.714.876-2.294.59-.581 1.375-.884 2.356-.91V4h.899v.996c.981.051 1.746.357 2.295.917.549.555.85 1.338.906 2.35h-2.054c-.035-.483-.148-.85-.34-1.105-.186-.26-.455-.404-.807-.436v2.832c1.243.399 2.119.855 2.627 1.369.509.513.763 1.19.763 2.03 0 .99-.297 1.781-.891 2.372-.589.591-1.422.936-2.5 1.035v1.478h-.898m1.02 14.04v-4.769l-2.938 4.768h2.938M169.907 36v-2.403h-4.56v-1.96l3.986-6.457h2.68v6.643h1.254v1.774h-1.253V36h-2.107"/><path id="text5151" fill="#181717" fill-rule="nonzero" d="M213.66 12.728c0 .354.119.653.356.897.237.238.529.357.876.357.337 0 .627-.122.869-.365a1.23 1.23 0 00.363-.889c0-.339-.121-.627-.363-.865a1.166 1.166 0 00-.869-.365c-.347 0-.64.12-.876.357a1.173 1.173 0 00-.356.873m-1.713 0c0-.825.284-1.52.853-2.087.568-.566 1.266-.849 2.092-.849.81 0 1.5.289 2.07.865.573.571.86 1.262.86 2.071 0 .815-.287 1.513-.86 2.095a2.814 2.814 0 01-2.07.865c-.826 0-1.524-.286-2.092-.857-.569-.572-.853-1.273-.853-2.103m-4.588 2.96L214.103 4h1.27l-6.743 11.688h-1.271m-.695-8.728c0 .349.116.642.347.88.237.238.53.357.877.357.342 0 .632-.119.869-.357.242-.243.363-.537.363-.88 0-.339-.121-.627-.363-.865a1.166 1.166 0 00-.869-.365c-.347 0-.64.119-.877.357a1.187 1.187 0 00-.347.873m-1.714 0c0-.826.285-1.524.853-2.095.569-.577 1.264-.865 2.085-.865.81 0 1.5.29 2.069.873a2.85 2.85 0 01.86 2.087 2.85 2.85 0 01-.86 2.086 2.81 2.81 0 01-2.069.857c-.827 0-1.524-.283-2.093-.849-.563-.566-.845-1.264-.845-2.094m2.259 25.66h2.21c.038.497.204.883.498 1.158.295.27.69.405 1.185.405.61 0 1.076-.18 1.398-.54.326-.36.49-.875.49-1.547 0-.63-.167-1.124-.498-1.484-.332-.365-.79-.547-1.374-.547-.327 0-.614.066-.861.198a1.757 1.757 0 00-.632.587l-2.037-.119.734-6.093h6.5v1.92h-4.905l-.292 2.277c.248-.196.532-.341.853-.436.321-.1.68-.151 1.074-.151 1.127 0 2.035.344 2.724 1.031.695.688 1.043 1.59 1.043 2.706 0 1.222-.382 2.198-1.145 2.928-.764.725-1.788 1.087-3.072 1.087-1.169 0-2.1-.299-2.795-.897-.69-.603-1.056-1.43-1.098-2.483"/><path id="text5157" fill="#181717" fill-rule="nonzero" d="M248.477 4h2.056l3.922 4.418h-1.916l-3.026-2.922-3.042 2.922h-1.917L248.477 4m-.685 28.078c0 .646.166 1.158.498 1.536.337.378.794.567 1.37.567.538 0 .96-.184 1.264-.552.305-.367.457-.88.457-1.535 0-.604-.158-1.074-.473-1.41-.316-.336-.759-.504-1.33-.504-.56 0-.997.168-1.313.504-.315.336-.473.8-.473 1.394m-.032-2.946a2.55 2.55 0 01.962-.59 3.652 3.652 0 011.256-.205c1.142 0 2.055.336 2.74 1.008.69.672 1.036 1.567 1.036 2.686 0 1.186-.375 2.144-1.125 2.874S250.889 36 249.659 36c-1.418 0-2.497-.473-3.237-1.418-.734-.945-1.1-2.328-1.1-4.15 0-1.959.393-3.455 1.182-4.49.788-1.04 1.924-1.559 3.409-1.559 1.033 0 1.865.26 2.495.78.637.52.984 1.223 1.044 2.11h-2.365c-.054-.362-.201-.635-.44-.819-.24-.183-.574-.275-1.003-.275-.588 0-1.039.246-1.354.74-.316.488-.492 1.226-.53 2.213"/><path id="text5163" fill="#181717" fill-rule="nonzero" d="M288.584 10.586l-.571.436c-.23.17-.404.373-.519.611a1.815 1.815 0 00-.173.801c0 .45.146.828.436 1.135.29.307.654.46 1.09.46.321 0 .627-.069.917-.206.296-.143.57-.35.82-.619l-2-2.618m.602-2.5l.263-.198c.24-.175.418-.37.533-.587.116-.217.173-.466.173-.746 0-.27-.08-.484-.24-.643-.156-.158-.369-.238-.64-.238-.275 0-.493.082-.653.246-.16.159-.24.376-.24.65 0 .149.037.31.112.485.08.174.193.36.338.555l.354.476m-1.692 1.056l-.436-.603a3.832 3.832 0 01-.489-.89 2.72 2.72 0 01-.142-.88c0-.846.255-1.518.766-2.015.517-.503 1.216-.754 2.098-.754.847 0 1.516.23 2.007.69.491.46.737 1.082.737 1.865 0 .592-.145 1.124-.436 1.595-.286.465-.737.904-1.353 1.317l1.586 2.063c.15-.26.268-.55.353-.873a5.97 5.97 0 00.188-1.079h1.955a7.196 7.196 0 01-.406 1.912 6 6 0 01-.887 1.61l2.015 2.588h-2.557l-.751-.976a4.672 4.672 0 01-1.459.976 4.299 4.299 0 01-1.654.325c-1.022 0-1.86-.32-2.51-.96-.647-.64-.97-1.457-.97-2.452 0-.74.18-1.378.54-1.912.362-.534.963-1.05 1.805-1.547M289.742 36h-2.248c.08-1.55.434-3.087 1.06-4.61.632-1.523 1.531-3.02 2.699-4.49h-5.293v-1.937h7.631v1.706c-1.183 1.375-2.092 2.83-2.729 4.364-.631 1.534-1.005 3.19-1.12 4.967"/><path id="text5169" fill="#181717" fill-rule="nonzero" d="M328.705 8.587l-1.353 1.948-1.308-.935 1.477-1.893-2.153-.612.5-1.595 2.068.754V4h1.523v2.254l2.068-.738.516 1.594-2.154.597 1.462 1.878-1.339.982-1.307-1.98m-1.722 23.823c0 .55.148.974.446 1.273.302.298.733.448 1.292.448.533 0 .95-.152 1.253-.456.308-.304.461-.725.461-1.265 0-.518-.156-.934-.469-1.248-.312-.315-.728-.472-1.245-.472-.518 0-.939.16-1.262.48-.317.319-.476.732-.476 1.24m.176-4.822c0 .434.134.77.4 1.005.267.23.649.346 1.146.346.497 0 .88-.118 1.146-.354.272-.235.408-.568.408-.997 0-.42-.139-.75-.416-.99-.277-.24-.656-.361-1.138-.361-.471 0-.848.123-1.13.369-.277.246-.416.573-.416.982m-.792 2.183c-.461-.262-.794-.568-1-.919-.204-.35-.307-.798-.307-1.343 0-.963.323-1.72.969-2.27.646-.55 1.538-.825 2.676-.825 1.149 0 2.046.275 2.692.825.65.545.976 1.301.976 2.27 0 .518-.11.969-.33 1.351-.22.377-.551.68-.992.911.538.257.943.613 1.215 1.068.271.45.407 1 .407 1.65 0 1.094-.353 1.953-1.061 2.576-.702.623-1.671.935-2.907.935-1.25 0-2.222-.309-2.914-.927-.692-.618-1.039-1.48-1.039-2.584 0-.644.129-1.183.385-1.618.261-.44.672-.807 1.23-1.1"/><path id="text5175" fill="#181717" fill-rule="nonzero" d="M372.099 4a14.885 14.885 0 00-1.566 3.611 14.3 14.3 0 00-.512 3.855c0 1.34.17 2.622.512 3.847a14.884 14.884 0 001.566 3.611h-1.612c-.891-1.25-1.568-2.507-2.032-3.768-.458-1.267-.687-2.497-.687-3.69s.229-2.423.687-3.69c.464-1.266 1.14-2.525 2.032-3.776h1.612m-1.169 24.33c0-.644-.158-1.152-.474-1.523-.31-.377-.735-.566-1.275-.566-.51 0-.907.181-1.192.542-.28.361-.42.866-.42 1.515 0 .607.148 1.081.443 1.421.296.34.71.51 1.245.51.525 0 .935-.167 1.23-.502.295-.335.443-.8.443-1.397m.038 2.928a2.45 2.45 0 01-.901.589c-.346.13-.736.196-1.169.196-1.074 0-1.935-.332-2.581-.997-.647-.665-.97-1.554-.97-2.67 0-1.182.35-2.137 1.054-2.865.702-.727 1.63-1.091 2.78-1.091 1.334 0 2.347.474 3.04 1.421.698.942 1.046 2.319 1.046 4.13 0 1.947-.371 3.438-1.115 4.475-.743 1.036-1.813 1.554-3.208 1.554-.962 0-1.739-.256-2.33-.77-.59-.518-.914-1.221-.97-2.111l2.208.008c.05.36.188.63.412.808.224.178.54.267.947.267.55 0 .97-.246 1.26-.738.291-.492.456-1.227.497-2.206"/><path id="text5181" fill="#181717" fill-rule="nonzero" d="M406.175 4h1.63c.888 1.247 1.561 2.504 2.02 3.772.46 1.263.69 2.497.69 3.701 0 1.2-.23 2.434-.69 3.702-.459 1.268-1.132 2.522-2.02 3.764h-1.63a14.864 14.864 0 001.576-3.623c.347-1.23.52-2.512.52-3.843 0-1.335-.173-2.616-.52-3.842A15.002 15.002 0 00406.175 4m1.048 26.224c0 1.367.133 2.36.398 2.978.27.619.7.928 1.286.928.587 0 1.015-.312 1.286-.936.27-.623.405-1.613.405-2.97 0-1.362-.135-2.352-.405-2.97-.27-.619-.7-.928-1.286-.928-.587 0-1.015.31-1.286.927-.265.613-.398 1.604-.398 2.971m-2.273 0c0-1.933.327-3.38.98-4.338.658-.964 1.65-1.446 2.977-1.446 1.321 0 2.311.485 2.97 1.454.663.964.994 2.412.994 4.346 0 1.933-.329 3.376-.987 4.33-.653.953-1.645 1.43-2.977 1.43-1.326 0-2.319-.48-2.977-1.438-.653-.959-.98-2.405-.98-4.338"/><path id="text5187" fill="#181717" fill-rule="nonzero" d="M444.554 18h7.921v1.634h-7.92V18m1.786 15v-2.194h4.347V33h-4.347"/><path id="text5201" fill="#181717" fill-rule="nonzero" d="M488.853 6h1.502v3.952h3.704v1.543h-3.704v3.967h-1.502v-3.967h-3.704V9.952h3.704V6m-3.704 25.458h8.91V33h-8.91v-1.542m0-3.333h8.91v1.543h-8.91v-1.543"/><path id="text5427" fill="#181717" fill-rule="nonzero" d="M7.278 63.794v-6.402H4.95V56h6.253v1.392H8.892v6.402H7.278m8.037-.715c-.195.3-.45.53-.767.688a2.472 2.472 0 01-1.102.233c-.561 0-.995-.159-1.3-.476-.302-.318-.453-.77-.453-1.355 0-.543.151-.957.453-1.243.305-.286.808-.482 1.507-.587.16-.025.37-.053.629-.085.657-.085.985-.273.985-.566 0-.233-.073-.399-.218-.498-.146-.102-.389-.153-.73-.153-.312 0-.552.062-.719.185a.619.619 0 00-.25.53v.084H11.9v-.106c0-.589.21-1.051.629-1.386.419-.339.997-.508 1.736-.508.81 0 1.43.14 1.859.418.433.279.65.68.65 1.206v3.27c0 .24.024.42.074.54a.52.52 0 00.25.264v.26h-1.613a1.337 1.337 0 01-.123-.328 1.884 1.884 0 01-.048-.387m-.026-2.116c-.253.116-.544.21-.874.28-.327.07-.495.108-.506.111-.273.078-.465.175-.575.292-.107.116-.16.276-.16.481 0 .212.07.38.208.508.138.123.325.185.559.185.419 0 .747-.116.985-.35.242-.235.363-.556.363-.962v-.545M18.165 56h1.507v2.757c.192-.3.431-.526.719-.678.291-.155.621-.232.99-.232.7 0 1.272.287 1.716.862.443.571.665 1.316.665 2.233 0 .91-.222 1.645-.665 2.206-.444.561-1.023.841-1.737.841-.373 0-.697-.077-.974-.232-.274-.156-.522-.403-.746-.741v.778h-1.475V56m4.058 4.878c0-.529-.112-.943-.335-1.243a1.082 1.082 0 00-.922-.455c-.412 0-.731.148-.959.444-.223.293-.335.711-.335 1.254 0 .586.106 1.03.32 1.334.216.303.53.455.942.455.42 0 .739-.152.959-.455.22-.307.33-.752.33-1.334"/><path id="text5207" fill="#181717" fill-rule="nonzero" d="M526.261 29.323h1.868c.44 0 .762-.08.963-.241.201-.16.302-.414.302-.76 0-.36-.1-.618-.297-.775-.198-.16-.524-.241-.978-.241h-1.858v2.017m0-3.337h1.8c.39 0 .674-.066.854-.2.18-.135.27-.348.27-.638 0-.283-.088-.487-.265-.613-.173-.13-.463-.194-.869-.194h-1.79v1.645m-1.509 4.73V23h3.611c.784 0 1.381.168 1.79.503.413.332.62.815.62 1.45 0 .392-.077.719-.23.98a1.51 1.51 0 01-.686.608c.395.16.69.393.884.697.198.3.297.676.297 1.126 0 .75-.243 1.33-.729 1.74-.485.408-1.174.612-2.065.612h-3.492m10.625-.707c-.19.297-.44.524-.75.68-.308.155-.667.231-1.076.231-.548 0-.971-.157-1.27-.471-.295-.314-.442-.761-.442-1.341 0-.538.147-.948.442-1.231.299-.283.79-.477 1.473-.582.156-.024.36-.052.614-.083.641-.084.962-.271.962-.56 0-.231-.07-.395-.213-.493-.142-.102-.38-.152-.713-.152-.305 0-.54.06-.702.183-.163.122-.245.297-.245.524v.084h-1.415v-.105c0-.583.205-1.04.614-1.372.41-.336.975-.503 1.696-.503.791 0 1.396.138 1.816.414.423.275.635.674.635 1.194v3.237c0 .238.024.416.073.534.048.116.13.203.244.262v.257h-1.576a1.334 1.334 0 01-.12-.325 1.89 1.89 0 01-.047-.382m-.026-2.095c-.246.115-.53.207-.853.277-.32.07-.484.107-.494.11-.268.077-.455.173-.562.288-.104.116-.156.274-.156.477 0 .21.067.377.202.503.136.122.318.183.547.183.41 0 .73-.115.962-.346.236-.233.354-.551.354-.953v-.54m6.197.655h1.493c-.062.723-.324 1.294-.785 1.713-.462.42-1.055.629-1.78.629-.822 0-1.467-.269-1.935-.807-.465-.541-.698-1.288-.698-2.242 0-.95.238-1.694.713-2.231.479-.542 1.136-.812 1.972-.812.732 0 1.318.195 1.759.586.444.392.692.936.744 1.635h-1.504c-.041-.304-.147-.536-.317-.697-.17-.16-.394-.24-.671-.24-.375 0-.656.148-.843.444-.184.297-.276.742-.276 1.336 0 .548.095.971.286 1.268a.94.94 0 00.833.44c.28 0 .506-.086.676-.257.17-.171.281-.426.333-.765m2.482 2.148V23h1.472v4.05l1.8-2.028h1.822l-1.972 2.111 2.091 3.583h-1.815l-1.353-2.425-.573.623v1.802h-1.472m5.562-1.854h1.509c.014.29.118.506.312.65.194.139.486.209.874.209.295 0 .522-.05.682-.147a.475.475 0 00.244-.424c0-.259-.301-.452-.905-.582a13.134 13.134 0 01-.588-.136c-.735-.188-1.247-.412-1.535-.67-.284-.259-.427-.608-.427-1.048 0-.583.209-1.044.625-1.383.42-.342.987-.513 1.701-.513.756 0 1.35.17 1.78.508.433.339.664.815.692 1.43h-1.473a.706.706 0 00-.276-.56c-.17-.133-.407-.2-.712-.2-.288 0-.503.044-.646.131a.434.434 0 00-.208.393c0 .227.377.428 1.13.603.173.038.31.07.41.094.753.178 1.267.393 1.54.644.278.252.417.606.417 1.064 0 .646-.227 1.138-.682 1.477-.45.339-1.108.508-1.972.508-.78 0-1.39-.176-1.831-.53-.44-.352-.661-.837-.661-1.455v-.063m10.14-.959c0-.573-.107-1.01-.322-1.31-.215-.303-.527-.455-.936-.455-.403 0-.71.15-.921.45-.208.3-.312.739-.312 1.315 0 .538.109.953.327 1.247.222.293.534.44.937.44.381 0 .681-.15.9-.45.219-.301.328-.713.328-1.237M555.768 33v-7.978h1.441v.77c.219-.335.462-.58.729-.733.27-.154.588-.23.952-.23.697 0 1.262.277 1.696.832.434.552.65 1.278.65 2.18 0 .91-.216 1.651-.65 2.22-.434.566-.992.849-1.675.849-.361 0-.684-.077-.968-.23a1.927 1.927 0 01-.702-.671V33h-1.473m9.745-2.991c-.19.297-.44.524-.749.68-.309.155-.668.231-1.077.231-.548 0-.971-.157-1.27-.471-.294-.314-.442-.761-.442-1.341 0-.538.148-.948.443-1.231.298-.283.789-.477 1.472-.582.156-.024.36-.052.614-.083.642-.084.963-.271.963-.56 0-.231-.072-.395-.214-.493-.142-.102-.38-.152-.713-.152-.305 0-.539.06-.702.183-.163.122-.245.297-.245.524v.084h-1.415v-.105c0-.583.205-1.04.614-1.372.41-.336.975-.503 1.696-.503.791 0 1.397.138 1.816.414.423.275.635.674.635 1.194v3.237c0 .238.024.416.073.534.048.116.13.203.244.262v.257h-1.576a1.334 1.334 0 01-.12-.325 1.89 1.89 0 01-.047-.382m-.026-2.095c-.246.115-.53.207-.853.277-.319.07-.484.107-.494.11-.267.077-.455.173-.562.288-.104.116-.156.274-.156.477 0 .21.067.377.203.503.135.122.317.183.546.183.41 0 .73-.115.963-.346.235-.233.353-.551.353-.953v-.54m6.197.655h1.494c-.063.723-.325 1.294-.786 1.713-.461.42-1.055.629-1.78.629-.822 0-1.467-.269-1.935-.807-.465-.541-.697-1.288-.697-2.242 0-.95.237-1.694.712-2.231.48-.542 1.136-.812 1.972-.812.732 0 1.319.195 1.76.586.443.392.691.936.743 1.635h-1.504c-.041-.304-.147-.536-.317-.697-.17-.16-.394-.24-.671-.24-.375 0-.656.148-.843.444-.184.297-.276.742-.276 1.336 0 .548.096.971.286 1.268a.94.94 0 00.833.44c.28 0 .506-.086.676-.257.17-.171.281-.426.333-.765m5.937.42h1.509c-.153.6-.461 1.072-.926 1.414-.462.339-1.025.508-1.691.508-.819 0-1.468-.276-1.946-.828-.479-.555-.718-1.306-.718-2.252 0-.933.236-1.668.707-2.206.472-.537 1.117-.806 1.936-.806.867 0 1.537.265 2.008.796.472.527.708 1.28.708 2.258a4.312 4.312 0 01-.016.398h-3.777c.02.447.128.784.322 1.01.198.228.48.341.849.341.26 0 .473-.05.64-.152.166-.104.298-.265.395-.482m-2.206-1.718h2.227c-.014-.384-.116-.675-.307-.875-.188-.202-.458-.303-.812-.303-.33 0-.59.1-.78.303-.188.203-.297.495-.328.875"/><path id="path5439" stroke="#181717" stroke-width="3" d="M560.396 11.5h-28.713"/><path id="path5441" fill="#181717" d="M534.653 17V7l-9.9 5z"/><g id="g5459" transform="translate(29.703 50)"><path id="path5453" stroke="#181717" stroke-width="3" d="M25.401 5.5H6.611"/><path id="path5455" fill="#181717" d="M10.556 10.656V.344L.66 5.5z"/><path id="path5457" stroke="#181717" stroke-width="3" d="M.66.344v10.312"/></g><g id="g5464" transform="rotate(180 27.723 35.5)"><path id="path5466" stroke="#181717" stroke-width="3" d="M25.401 5.5H6.611"/><path id="path5468" fill="#181717" d="M10.556 10.656V.344L.66 5.5z"/><path id="path5470" stroke="#181717" stroke-width="3" d="M.66.344v10.312"/></g><path id="text5474" fill="#181717" fill-rule="nonzero" d="M71.131 54.066l-1.178-1.225 1.254-1.386 1.208 1.257c.161-.322.285-.69.37-1.104.086-.414.129-.865.129-1.354 0-1.305-.275-2.318-.824-3.038-.548-.726-1.316-1.088-2.303-1.088-.977 0-1.737.36-2.281 1.08-.544.72-.816 1.735-.816 3.046 0 1.306.272 2.321.816 3.047.544.72 1.304 1.08 2.28 1.08.253 0 .49-.027.71-.08a2.74 2.74 0 00.635-.235m1.609 1.62a4.191 4.191 0 01-1.322.613 6.03 6.03 0 01-1.631.21c-1.662 0-2.983-.565-3.965-1.693-.977-1.128-1.466-2.649-1.466-4.562 0-1.918.489-3.439 1.466-4.562.982-1.128 2.303-1.692 3.965-1.692 1.661 0 2.983.564 3.965 1.692.987 1.129 1.48 2.65 1.48 4.562a8.3 8.3 0 01-.302 2.305 5.583 5.583 0 01-.89 1.822l1.208 1.233L74.008 57l-1.268-1.314"/><path id="text5482" fill="#181717" fill-rule="nonzero" d="M106.308 56l-3.338-12h2.483l1.993 8.464L109.131 44h2.554l1.684 8.464L115.362 44h2.46l-3.33 12h-2.285l-1.803-9.19-1.811 9.19h-2.285"/><path id="text5492" fill="#181717" fill-rule="nonzero" d="M145.545 56V44h8.62v2.086h-6.21v2.558h5.678v2.053h-5.677v3.087h6.5V56h-8.911"/><path id="text5500" fill="#181717" fill-rule="nonzero" d="M187.62 49.246h2.903c.63 0 1.093-.127 1.387-.382.299-.256.448-.658.448-1.206 0-.522-.144-.915-.432-1.181-.288-.272-.718-.408-1.289-.408h-3.017v3.177M185.149 56V44h5.823c1.31 0 2.28.28 2.911.84.63.559.946 1.414.946 2.565 0 .734-.155 1.344-.465 1.833a2.307 2.307 0 01-1.32 1.002c.521.185.89.467 1.108.848.223.38.351.972.384 1.776l.049 1.417v.049c.016.717.17 1.151.465 1.303V56h-2.708a2.362 2.362 0 01-.204-.62 6.444 6.444 0 01-.082-.887l-.032-1.263c-.027-.744-.169-1.246-.424-1.507-.25-.26-.704-.391-1.362-.391h-2.618V56h-2.471"/><path id="text5504" fill="#181717" fill-rule="nonzero" d="M229.06 56v-9.857h-3.317V44h8.91v2.143h-3.294V56h-2.3"/><path id="text5518" fill="#181717" fill-rule="nonzero" d="M271.618 56h-2.268v-4.473L265.347 44h2.776l2.357 5.173L272.658 44h2.59l-3.63 7.527V56"/><path id="text5522" fill="#181717" fill-rule="nonzero" d="M305.94 44h2.311v7.304c0 .902.172 1.56.514 1.977.342.41.886.616 1.631.616.755 0 1.304-.205 1.646-.616.348-.411.521-1.07.521-1.977V44h2.288v7.565c0 1.428-.382 2.525-1.147 3.289-.76.764-1.858 1.146-3.293 1.146-1.445 0-2.552-.38-3.323-1.138-.765-.765-1.147-1.863-1.147-3.297V44"/><path id="text5526" fill="#181717" fill-rule="nonzero" d="M349.505 56V44h1.98v12h-1.98"/><path id="text5530" fill="#181717" fill-rule="nonzero" d="M386.495 50c0 1.253.273 2.227.817 2.923.545.69 1.306 1.036 2.284 1.036.989 0 1.758-.346 2.307-1.036.55-.696.825-1.67.825-2.923 0-1.253-.275-2.224-.825-2.915-.55-.696-1.318-1.044-2.307-1.044-.978 0-1.74.346-2.284 1.036-.544.691-.817 1.665-.817 2.923m-2.337 0c0-1.84.49-3.299 1.468-4.376.983-1.083 2.306-1.624 3.97-1.624 1.664 0 2.988.541 3.971 1.624.988 1.082 1.483 2.541 1.483 4.376 0 1.835-.495 3.294-1.483 4.376-.983 1.083-2.307 1.624-3.97 1.624-1.665 0-2.988-.541-3.971-1.624-.978-1.082-1.468-2.54-1.468-4.376"/><path id="text5534" fill="#181717" fill-rule="nonzero" d="M428.156 49.556h2.319c.628 0 1.084-.136 1.37-.407.284-.277.426-.72.426-1.328 0-.57-.14-1.002-.419-1.296-.28-.293-.696-.44-1.25-.44h-2.446v3.471m-.016 2.175V56h-2.397V44h5.183c1.224 0 2.15.329 2.778.986.633.651.95 1.61.95 2.876 0 1.238-.31 2.194-.926 2.867-.618.668-1.499 1.002-2.644 1.002h-2.944"/><path id="text5546" fill="#181717" fill-rule="nonzero" d="M473.267 43.016v1.634c-.049 0-.117-.002-.205-.008a3.648 3.648 0 00-.18-.007c-.552 0-.926.101-1.122.305-.192.198-.287.62-.287 1.267v1.877c0 .771-.123 1.319-.369 1.642-.246.324-.688.556-1.327.696.639.141 1.081.37 1.327.689.246.318.369.863.369 1.634v1.885c0 .642.095 1.061.287 1.26.19.198.565.297 1.122.297.033 0 .093-.003.18-.008.088-.005.156-.008.205-.008v1.635c-.076 0-.183.002-.32.007a8.326 8.326 0 01-1.672-.086 3.288 3.288 0 01-.933-.289 1.51 1.51 0 01-.664-.72c-.125-.297-.188-.816-.188-1.556v-1.97c0-.715-.14-1.221-.418-1.518-.279-.303-.748-.454-1.41-.454-.032 0-.087.003-.163.008a2.764 2.764 0 01-.172.008V49.6c.038 0 .095.003.172.008.076.005.13.008.164.008.655 0 1.122-.151 1.4-.454.285-.302.427-.813.427-1.533v-1.955c0-.746.063-1.27.188-1.572a1.51 1.51 0 01.664-.72c.257-.13.568-.227.934-.29a8.326 8.326 0 011.672-.085c.136.005.243.008.32.008m-4.302 20.053h4.137v1.666h-1.925v11.6h1.925V78h-4.137V63.07"/><path id="text5554" fill="#181717" fill-rule="nonzero" d="M506.93 43h.558c.6 0 1.084.031 1.45.094.366.063.672.162.918.297.311.167.53.407.655.72.132.308.197.824.197 1.55v1.971c0 .715.14 1.224.418 1.526.279.297.748.446 1.41.446.032 0 .087-.003.163-.008.077-.005.134-.008.172-.008v1.636h-.303c-.683 0-1.163.146-1.442.438-.279.292-.418.798-.418 1.518v1.972c0 .746-.065 1.27-.197 1.573a1.393 1.393 0 01-.655.704c-.257.13-.568.227-.934.29a8.322 8.322 0 01-1.672.085 8.903 8.903 0 00-.32-.007V56.16c.05 0 .118.003.206.008.087.005.147.008.18.008.552 0 .923-.102 1.114-.305.197-.199.295-.616.295-1.252v-1.87c0-.783.123-1.333.369-1.651.246-.324.688-.553 1.327-.689-.639-.14-1.081-.373-1.327-.696-.246-.324-.369-.872-.369-1.644v-1.893c0-.637-.098-1.054-.295-1.252-.191-.204-.562-.305-1.114-.305-.033 0-.093.002-.18.008a3.64 3.64 0 01-.205.007V43m4.375 20.063V78h-4.146v-1.667h1.934V64.73h-1.934v-1.666h4.146"/><path id="text5564" fill="#181717" fill-rule="nonzero" d="M549.198 42h1.611v16.137h-1.61V42m2.205 34l-3.88-13.403h1.057L552.475 76h-1.071"/><g id="g5770" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="13" font-weight="normal" transform="translate(2.97 83)"><g id="text5570" transform="translate(0 .166)"><text id="tspan5572"><tspan x=".102" y="14.34">Caps Lock</tspan></text></g></g><path id="text5641" fill="#181717" fill-rule="nonzero" d="M79.684 86.403l-1.582 5.108h3.17l-1.588-5.108M78.319 84h2.713l4.117 12h-2.466l-.775-2.468h-4.426L76.73 96h-2.473l4.062-12"/><path id="text5645" fill="#181717" fill-rule="nonzero" d="M115.842 92.17h2.224c.084.623.33 1.086.736 1.39.407.298.994.448 1.763.448.654 0 1.148-.121 1.48-.363.332-.242.498-.6.498-1.074 0-.69-.954-1.26-2.863-1.714a6.949 6.949 0 01-.298-.07c-1.021-.231-1.75-.491-2.187-.78a2.488 2.488 0 01-.885-1.034c-.203-.438-.305-.952-.305-1.544 0-1.107.362-1.954 1.086-2.541.724-.592 1.77-.888 3.14-.888 1.278 0 2.278.314 2.997.942.724.628 1.105 1.514 1.145 2.656h-2.164c-.04-.55-.243-.97-.61-1.258-.367-.289-.89-.433-1.57-.433-.59 0-1.046.121-1.368.363-.318.237-.476.577-.476 1.02 0 .602.622 1.052 1.867 1.35l.788.194c.798.211 1.364.376 1.696.494.337.119.63.25.878.394.446.257.78.6 1.004 1.027.223.422.334.93.334 1.521 0 1.184-.384 2.103-1.152 2.757-.769.649-1.853.973-3.25.973-1.38 0-2.46-.332-3.244-.996-.783-.664-1.205-1.609-1.264-2.834"/><path id="text5649" fill="#181717" fill-rule="nonzero" d="M157.8 93.833h1.926c1.115 0 1.921-.299 2.418-.896.502-.603.753-1.578.753-2.925 0-1.341-.232-2.327-.698-2.957-.465-.63-1.191-.945-2.18-.945h-2.22v7.723M155.447 96V84h4.574c1.791 0 3.125.497 4.003 1.49.882.995 1.324 2.502 1.324 4.522 0 1.097-.164 2.064-.492 2.9-.322.837-.793 1.51-1.411 2.02-.465.381-.994.655-1.585.824-.592.163-1.422.244-2.49.244h-3.923"/><path id="text5653" fill="#181717" fill-rule="nonzero" d="M196.04 96V84h7.92v2.086h-5.617v2.72h4.924v2.086h-4.924V96h-2.303"/><path id="text5657" fill="#181717" fill-rule="nonzero" d="M243.628 94.338c-.436.572-.934.992-1.493 1.26-.554.268-1.206.402-1.956.402-1.645 0-2.978-.552-4-1.655-1.017-1.108-1.526-2.561-1.526-4.36 0-1.815.509-3.266 1.526-4.354 1.016-1.087 2.371-1.631 4.064-1.631 1.474 0 2.677.343 3.609 1.028.931.68 1.485 1.619 1.66 2.815h-2.467c-.149-.593-.455-1.044-.918-1.353-.458-.31-1.057-.464-1.796-.464-.985 0-1.76.35-2.324 1.051-.559.696-.838 1.66-.838 2.892 0 1.237.292 2.206.878 2.907.585.701 1.39 1.052 2.411 1.052.767 0 1.408-.22 1.925-.657.516-.439.835-1.039.958-1.802h-2.667v-1.948h4.87v6.162h-1.62l-.296-1.345"/><path id="text5661" fill="#181717" fill-rule="nonzero" d="M275.248 96V84h2.479v4.473h4.942V84h2.48v12h-2.48v-5.32h-4.942V96h-2.48"/><path id="text5665" fill="#181717" fill-rule="nonzero" d="M316.832 91.324h2.155v1.41c0 .407.113.72.339.936.225.211.55.317.971.317.461 0 .782-.13.964-.389.181-.259.272-.745.272-1.458V84h2.23v8.251c0 .74-.042 1.284-.126 1.633-.078.343-.21.647-.397.911-.284.391-.674.69-1.17.896-.495.206-1.076.309-1.744.309-.613 0-1.155-.09-1.626-.27a2.949 2.949 0 01-1.177-.8 2.594 2.594 0 01-.537-.951c-.103-.354-.154-.935-.154-1.744v-.911"/><path id="text5669" fill="#181717" fill-rule="nonzero" d="M355.446 96V84h2.29v4.929L362.08 84h2.865l-4.444 4.855L365.347 96h-2.797l-3.628-5.49-1.186 1.27V96h-2.29"/><path id="text5673" fill="#181717" fill-rule="nonzero" d="M396.04 96V84h2.348v9.784h5.572V96h-7.92"/><path id="text5679" fill="#181717" fill-rule="nonzero" d="M96.04 136v-2.086l6.105-7.739h-5.983V124h8.788v2.086l-6.12 7.747h5.983V136H96.04"/><path id="text5683" fill="#181717" fill-rule="nonzero" d="M134.653 136l3.521-6.126-3.52-5.874h2.661l2.289 4.196 2.274-4.196h2.676l-3.52 5.857 3.52 6.143h-2.661l-2.289-4.155-2.289 4.155h-2.662"/><path id="text5687" fill="#181717" fill-rule="nonzero" d="M185.149 131.77c-.092 1.294-.616 2.323-1.573 3.086-.95.763-2.193 1.144-3.727 1.144-1.767 0-3.142-.526-4.125-1.577-.978-1.052-1.467-2.526-1.467-4.423 0-1.938.5-3.423 1.5-4.454.999-1.03 2.436-1.546 4.31-1.546 1.524 0 2.726.356 3.607 1.067.886.706 1.372 1.706 1.458 3h-2.423c-.102-.644-.378-1.134-.826-1.47-.449-.34-1.054-.51-1.815-.51-1.08 0-1.902.333-2.464.998s-.843 1.637-.843 2.915c0 1.237.279 2.193.835 2.869.562.675 1.359 1.012 2.39 1.012.746 0 1.357-.18 1.832-.54.475-.367.783-.89.924-1.57h2.407"/><path id="text5691" fill="#181717" fill-rule="nonzero" d="M218.72 136l-3.869-12h2.56l2.383 8.978 2.43-8.978h2.528l-3.823 12h-2.209"/><path id="text5695" fill="#181717" fill-rule="nonzero" d="M257.822 133.833h2.943c.694 0 1.2-.125 1.516-.375.317-.25.476-.643.476-1.181 0-.56-.156-.961-.468-1.206-.311-.25-.825-.374-1.54-.374h-2.927v3.136m0-5.19h2.836c.612 0 1.06-.103 1.344-.309.285-.212.427-.543.427-.994 0-.44-.14-.758-.418-.953-.274-.201-.73-.301-1.37-.301h-2.819v2.558M255.446 136v-12h5.688c1.235 0 2.174.26 2.82.782.65.516.975 1.268.975 2.257 0 .608-.12 1.116-.361 1.523-.24.407-.601.722-1.082.945.623.25 1.087.611 1.393 1.084.312.467.468 1.05.468 1.751 0 1.168-.383 2.07-1.148 2.705-.765.635-1.85.953-3.254.953h-5.5"/><path id="text5699" fill="#181717" fill-rule="nonzero" d="M295.05 136v-12h2.585l4.877 8.244V124h2.438v12h-2.552l-4.91-8.244V136h-2.438"/><path id="text5703" fill="#181717" fill-rule="nonzero" d="M333.663 136v-12h3.597l2.352 9.132 2.32-9.132h3.613v12h-2.28v-9.678l-2.4 9.678h-2.49l-2.432-9.678V136h-2.28"/><path id="text5725" fill="#181717" fill-rule="nonzero" d="M438.614 89.363V87h1.98v2.363h-1.98m0 5.974v-2.348h1.98v2.348h-1.98m.013 13.99v-2.362h1.967v2.363h-1.967m0 8.672v-.912c.366-.125.636-.318.81-.578.175-.26.262-.597.262-1.013v-.195h-1.085v-2.348h1.98v2.254c0 .78-.166 1.404-.497 1.872-.33.473-.82.78-1.47.92"/><path id="text5731" fill="#181717" fill-rule="nonzero" d="M480.71 85h1.468v4.301h-1.468V85m-2.492 0h1.468v4.301h-1.468V85m1.254 19.699h1.468V109h-1.468v-4.301"/><path id="text5737" fill="#181717" fill-rule="nonzero" d="M514.851 93.804V86h5.524v1.356h-3.979v1.664h3.638v1.335h-3.638v2.008h4.165v1.441h-5.71m6.775 0v-5.759h1.472v.683c.21-.279.458-.487.744-.625.286-.141.605-.212.956-.212.62 0 1.084.166 1.39.498.31.329.465.825.465 1.49v3.925h-1.503v-3.48c0-.414-.07-.709-.207-.886-.134-.176-.355-.264-.662-.264-.354 0-.635.109-.842.328-.206.215-.31.51-.31.885v3.417h-1.503m8.82.042l-.496.016c-.19.01-.31.016-.362.016-.575 0-.97-.11-1.183-.328-.21-.223-.315-.66-.315-1.309v-3.115h-.744v-1.081h.744v-1.573h1.488v1.573h.868v1.08h-.868v3.19c0 .152.032.253.098.302.065.046.2.07.403.07h.367v1.16m4.335-1.791h1.499c-.152.607-.459 1.084-.92 1.43-.458.343-1.018.514-1.68.514-.813 0-1.457-.279-1.932-.837-.475-.562-.713-1.321-.713-2.278 0-.943.234-1.687.703-2.23.468-.545 1.109-.817 1.922-.817.861 0 1.526.269 1.994.806.469.533.703 1.294.703 2.283a4.44 4.44 0 01-.015.403h-3.752c.02.452.128.793.32 1.022.197.23.478.345.843.345.258 0 .47-.052.635-.154.166-.106.297-.268.393-.487m-2.19-1.738h2.21c-.013-.389-.115-.684-.304-.885-.186-.205-.455-.307-.806-.307-.327 0-.586.102-.775.307-.186.205-.295.5-.326.885m4.764 3.486v-5.759h1.395v.985c.197-.395.426-.685.688-.868.261-.188.573-.281.935-.281.058 0 .103.002.134.005.035 0 .062.002.083.005l.005 1.563h-.501c-.41 0-.718.11-.925.329-.207.219-.31.544-.31.975v3.046h-1.504"/><path id="path5745" fill="#181717" d="M524.752 115v-10l-9.9 5z"/><path id="path5747" stroke="#181717" stroke-width="3" d="M539.604 100v10h-19.802"/><g id="g5790" transform="translate(6.901 120)"><text id="tspan5755" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x=".208" y="17.237">Shift</tspan></text><path id="path5757" stroke="#181717" stroke-width="2" d="M8.16 19.535l-7.494 10h4.996v5h4.996v-5h4.996z"/></g><g id="g5795" transform="translate(495.05 127)"><g id="text5797" fill="#181717" fill-rule="nonzero" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" transform="translate(30.402)"><text id="tspan5799"><tspan x=".537" y="17.955">Shift</tspan></text></g><path id="path5801" stroke="#181717" stroke-width="2" d="M7.55 10.628L0 20.58h5.033v4.976h5.034V20.58H15.1z"/></g><path id="text5803" fill="#181717" fill-rule="nonzero" d="M384.158 126v1.71l-6.755 2.848 6.755 2.841v1.71l-8.91-3.786v-1.553l8.91-3.77m-5.605 32v-.913c.424-.125.736-.318.938-.578.202-.26.303-.598.303-1.014v-.195h-1.24v-2.35h2.291v2.256c0 .775-.194 1.4-.582 1.873-.384.474-.953.78-1.71.921"/><path id="text5813" fill="#181717" fill-rule="nonzero" d="M415.842 127l8.91 3.731v1.538l-8.91 3.746v-1.692l6.77-2.812-6.77-2.82V127m3.305 29v-2.325h2.277V156h-2.277"/><path id="text5819" fill="#181717" fill-rule="nonzero" d="M459.343 132.387c0-.056-.002-.131-.008-.224a4.45 4.45 0 01-.008-.217c0-.454.05-.831.151-1.13.101-.305.263-.59.485-.853.17-.196.421-.42.755-.673.339-.253.559-.437.66-.55.211-.233.357-.442.436-.628.08-.185.12-.39.12-.611 0-.49-.141-.873-.422-1.147-.28-.273-.672-.41-1.176-.41-.503 0-.9.165-1.191.496-.286.325-.442.784-.469 1.378h-2.24v-.24c0-1.09.352-1.957 1.056-2.602.71-.65 1.663-.976 2.86-.976 1.224 0 2.196.307 2.916.922.726.609 1.088 1.427 1.088 2.455 0 .361-.042.686-.127.976-.08.283-.204.544-.373.782-.217.3-.554.624-1.009.976-.45.345-.723.56-.818.642a1.828 1.828 0 00-.437.604 2.04 2.04 0 00-.12.891 1.8 1.8 0 01.008.14h-2.137m-.11 3.314v-2.269h2.343v2.27h-2.344M457.897 157l3.941-13.174h1.088L458.97 157h-1.073"/><path id="text5827" fill="#181717" fill-rule="nonzero" d="M11.801 181.18c-.057.863-.387 1.549-.988 2.057-.599.509-1.38.763-2.345.763-1.112 0-1.976-.35-2.595-1.052-.615-.7-.923-1.683-.923-2.948 0-1.292.315-2.282.944-2.97.628-.686 1.532-1.03 2.711-1.03.959 0 1.715.237 2.269.711.557.471.863 1.138.917 2h-1.524c-.064-.43-.238-.756-.52-.979-.282-.227-.663-.34-1.142-.34-.68 0-1.196.221-1.55.665-.353.443-.53 1.09-.53 1.943 0 .825.176 1.462.526 1.912.353.45.854.676 1.503.676.47 0 .853-.12 1.152-.361.3-.244.493-.593.581-1.047h1.514m3.569 2.65c-.136.003-.3.008-.49.015-.187.01-.306.016-.357.016-.567 0-.956-.107-1.167-.32-.207-.216-.31-.64-.31-1.273v-3.03h-.735v-1.052h.734v-1.531h1.468v1.53h.857v1.052h-.857v3.103c0 .148.033.246.097.294.065.045.197.067.398.067h.362v1.129m.861-.041v-5.603h1.376v.958c.194-.385.42-.666.678-.845a1.56 1.56 0 01.923-.273c.058 0 .102.001.133.005.033 0 .06.002.081.005l.005 1.52h-.494c-.405 0-.709.107-.913.32-.204.213-.306.53-.306.949v2.964h-1.483m4.068 0v-7.593h1.483v7.593H20.3"/><path id="text5831" fill="#181717" fill-rule="nonzero" d="M108.714 177.587l-1.097 3.373h2.2l-1.103-3.373m-.946-1.587h1.882l2.855 7.925h-1.71l-.538-1.63h-3.07l-.521 1.63h-1.716l2.818-7.925m5.468 7.925V176h1.565v7.925h-1.565m5.576.043a49.21 49.21 0 00-.516.016c-.197.01-.323.016-.377.016-.598 0-1.009-.111-1.231-.334-.219-.226-.328-.668-.328-1.328v-3.164h-.774v-1.097h.774v-1.598h1.549v1.598h.903v1.097h-.903v3.239c0 .154.034.256.102.307.068.046.208.07.42.07h.38v1.178"/><path id="text5835" fill="#181717" fill-rule="nonzero" d="M408.714 177.587l-1.097 3.373h2.2l-1.103-3.373m-.946-1.587h1.882l2.855 7.925h-1.71l-.538-1.63h-3.07l-.521 1.63h-1.716l2.818-7.925m5.468 7.925V176h1.565v7.925h-1.565m5.576.043a49.21 49.21 0 00-.516.016c-.197.01-.323.016-.377.016-.598 0-1.009-.111-1.231-.334-.219-.226-.328-.668-.328-1.328v-3.164h-.774v-1.097h.774v-1.598h1.549v1.598h.903v1.097h-.903v3.239c0 .154.034.256.102.307.068.046.208.07.42.07h.38v1.178"/><path id="text5839" fill="#181717" fill-rule="nonzero" d="M552.395 181.18c-.057.863-.387 1.549-.988 2.057-.599.509-1.38.763-2.345.763-1.111 0-1.976-.35-2.595-1.052-.615-.7-.922-1.683-.922-2.948 0-1.292.314-2.282.943-2.97.628-.686 1.532-1.03 2.711-1.03.959 0 1.715.237 2.269.711.557.471.863 1.138.917 2h-1.524c-.064-.43-.238-.756-.52-.979-.282-.227-.662-.34-1.142-.34-.68 0-1.196.221-1.55.665-.353.443-.53 1.09-.53 1.943 0 .825.176 1.462.526 1.912.353.45.854.676 1.503.676.47 0 .853-.12 1.152-.361.3-.244.493-.593.582-1.047h1.513m3.569 2.65c-.136.003-.3.008-.49.015-.187.01-.306.016-.356.016-.568 0-.957-.107-1.168-.32-.207-.216-.31-.64-.31-1.273v-3.03h-.735v-1.052h.734v-1.531h1.468v1.53h.857v1.052h-.857v3.103c0 .148.033.246.097.294.065.045.197.067.398.067h.362v1.129m.861-.041v-5.603h1.376v.958c.194-.385.42-.666.678-.845a1.56 1.56 0 01.923-.273c.058 0 .102.001.133.005.034 0 .06.002.081.005l.005 1.52h-.494c-.405 0-.709.107-.913.32-.204.213-.306.53-.306.949v2.964h-1.483m4.068 0v-7.593h1.483v7.593h-1.483"/><path id="text5872" fill="#181717" fill-rule="nonzero" d="M72.458 176.567L70.297 169h1.608l1.29 5.338 1.09-5.338h1.655l1.09 5.338L78.32 169h1.593l-2.156 7.567h-1.48l-1.167-5.795-1.172 5.795h-1.48m8.228-6.206V169h1.49v1.361h-1.49m0 6.206v-5.584h1.49v5.584h-1.49m2.816 0v-5.584h1.46v.663c.208-.27.453-.473.737-.606.283-.137.599-.206.947-.206.614 0 1.074.161 1.377.483.308.318.461.8.461 1.444v3.806h-1.49v-3.375c0-.4-.068-.687-.205-.858-.133-.171-.351-.257-.655-.257-.352 0-.63.106-.835.319-.204.209-.307.495-.307.858v3.313h-1.49m-12.55 13.152v-7.567h1.552v3.108l2.944-3.108h1.94l-3.01 3.062 3.282 4.505h-1.895l-2.457-3.463-.804.802v2.661h-1.552m10.676-1.695h1.485c-.15.589-.454 1.051-.911 1.387-.454.332-1.009.498-1.664.498-.806 0-1.444-.27-1.915-.812-.471-.544-.707-1.28-.707-2.209 0-.914.232-1.635.696-2.163.465-.527 1.1-.79 1.905-.79.853 0 1.512.26 1.977.78.464.517.696 1.255.696 2.214a4.215 4.215 0 01-.015.39h-3.718c.02.44.126.77.318.992.194.223.472.334.834.334.256 0 .466-.05.63-.149.164-.102.294-.26.39-.472m-2.172-1.685h2.192c-.014-.377-.115-.663-.302-.858-.185-.199-.451-.298-.8-.298-.323 0-.58.099-.767.298-.185.198-.292.484-.323.858m4.941 5.62v-1.207a1.976 1.976 0 00.348.02c.294 0 .514-.07.661-.21.147-.137.22-.345.22-.622a.799.799 0 00-.03-.154l-2.008-5.651h1.639l1.172 4.027 1.142-4.027h1.567l-2.289 6.586c-.17.493-.384.829-.64 1.007-.256.181-.64.272-1.152.272-.092 0-.191-.003-.297-.01a6.338 6.338 0 01-.333-.031"/><path id="text5878" fill="#181717" fill-rule="nonzero" d="M472.458 176.567L470.297 169h1.608l1.29 5.338 1.09-5.338h1.655l1.09 5.338 1.29-5.338h1.593l-2.156 7.567h-1.48l-1.167-5.795-1.172 5.795h-1.48m8.228-6.206V169h1.49v1.361h-1.49m0 6.206v-5.584h1.49v5.584h-1.49m2.816 0v-5.584h1.46v.663c.208-.27.453-.473.737-.606.283-.137.599-.206.947-.206.614 0 1.074.161 1.377.483.308.318.461.8.461 1.444v3.806h-1.49v-3.375c0-.4-.068-.687-.205-.858-.133-.171-.351-.257-.655-.257-.352 0-.63.106-.835.319-.204.209-.307.495-.307.858v3.313h-1.49m-12.55 13.152v-7.567h1.552v3.108l2.944-3.108h1.94l-3.01 3.062 3.282 4.505h-1.895l-2.457-3.463-.804.802v2.661h-1.552m10.676-1.695h1.485c-.15.589-.454 1.051-.911 1.387-.454.332-1.009.498-1.664.498-.806 0-1.444-.27-1.915-.812-.471-.544-.707-1.28-.707-2.209 0-.914.232-1.635.696-2.163.465-.527 1.1-.79 1.905-.79.853 0 1.512.26 1.977.78.464.517.696 1.255.696 2.214a4.215 4.215 0 01-.015.39h-3.718c.02.44.126.77.318.992.194.223.472.334.834.334.256 0 .466-.05.63-.149.164-.102.294-.26.39-.472m-2.172-1.685h2.192c-.014-.377-.115-.663-.302-.858-.185-.199-.451-.298-.8-.298-.323 0-.58.099-.767.298-.185.198-.292.484-.323.858m4.941 5.62v-1.207a1.976 1.976 0 00.348.02c.294 0 .514-.07.661-.21.147-.137.22-.345.22-.622a.8.8 0 00-.03-.154l-2.008-5.651h1.639l1.172 4.027 1.142-4.027h1.567l-2.289 6.586c-.17.493-.384.829-.64 1.007-.256.181-.64.272-1.152.272-.092 0-.191-.003-.297-.01a6.338 6.338 0 01-.333-.031"/><path id="text5884" fill="#181717" fill-rule="nonzero" d="M506.93 183.804V176h2.338l1.529 5.94 1.507-5.94h2.348v7.804h-1.482v-6.294l-1.56 6.294h-1.617l-1.58-6.294v6.294h-1.482m12.693-1.748h1.513c-.153.607-.463 1.084-.929 1.43-.462.343-1.027.514-1.695.514-.821 0-1.471-.279-1.951-.837-.48-.562-.72-1.321-.72-2.278 0-.943.236-1.687.71-2.23.472-.545 1.12-.817 1.94-.817.87 0 1.54.269 2.014.806.473.533.71 1.294.71 2.283a4.4 4.4 0 01-.016.403h-3.788c.02.452.129.793.323 1.022.199.23.482.345.85.345.262 0 .476-.052.643-.154.166-.106.299-.268.396-.487m-2.212-1.738h2.233c-.014-.389-.117-.684-.308-.885-.188-.205-.459-.307-.814-.307-.33 0-.591.102-.782.307-.188.205-.298.5-.33.885m4.811 3.486v-5.759h1.487v.683a2.05 2.05 0 01.751-.625c.289-.141.61-.212.965-.212.627 0 1.094.166 1.404.498.313.329.47.825.47 1.49v3.925h-1.519v-3.48c0-.414-.07-.709-.208-.886-.136-.176-.359-.264-.668-.264-.358 0-.642.109-.85.328-.21.215-.314.51-.314.885v3.417h-1.518m11.441-5.759v5.759h-1.487v-.683c-.215.279-.467.489-.756.63a2.191 2.191 0 01-.965.207c-.623 0-1.09-.166-1.404-.498-.31-.332-.464-.829-.464-1.49v-3.925h1.518v3.48c0 .41.068.704.204.88.135.173.36.26.673.26.355 0 .636-.108.845-.323.212-.22.318-.516.318-.89v-3.407h1.518"/><circle id="Oval" cx="109.168" cy="139" r="26" stroke="#C06334" stroke-width="5"/></g></g></svg> \ No newline at end of file diff --git a/2-ui/3-event-details/8-onscroll/1-endless-page/solution.md b/2-ui/3-event-details/8-onscroll/1-endless-page/solution.md index f1f4bd85b..54c101193 100644 --- a/2-ui/3-event-details/8-onscroll/1-endless-page/solution.md +++ b/2-ui/3-event-details/8-onscroll/1-endless-page/solution.md @@ -8,12 +8,12 @@ Let's use window-relative coordinates. The document is represented (and contained) within `<html>` tag, that is `document.documentElement`. -We can get window-relative coordinates of the whole document as `document.documentElement.getBoundingClientRect()`. And the `bottom` property will be window-relative coordinate of the document end. +We can get window-relative coordinates of the whole document as `document.documentElement.getBoundingClientRect()`, the `bottom` property will be window-relative coordinate of the document bottom. -For instance, if the height of the whole HTML document is 2000px, then: +For instance, if the height of the whole HTML document is `2000px`, then: ```js -// When we're on the top of the page +// when we're on the top of the page // window-relative top = 0 document.documentElement.getBoundingClientRect().top = 0 @@ -41,11 +41,11 @@ document.documentElement.getBoundingClientRect().top = -1400 document.documentElement.getBoundingClientRect().bottom = 600 ``` -Please note that the bottom can't be 0, because it never reaches the window top. The lowest limit of the bottom coordinate is the window height, we can't scroll it any more up. +Please note that the `bottom` can't be `0`, because it never reaches the window top. The lowest limit of the `bottom` coordinate is the window height (we assumed it to be `600`), we can't scroll it any more up. -And the window height is `document.documentElement.clientHeight`. +We can obtain the window height as `document.documentElement.clientHeight`. -We want the document bottom be no more than `100px` away from it. +For our task, we need to know when the document bottom is not no more than `100px` away from it (that is: `600-700px`, if the height is `600`). So here's the function: @@ -55,11 +55,10 @@ function populate() { // document bottom let windowRelativeBottom = document.documentElement.getBoundingClientRect().bottom; - // if it's greater than window height + 100px, then we're not at the page back - // (see examples above, big bottom means we need to scroll more) + // if the user hasn't scrolled far enough (>100px to the end) if (windowRelativeBottom > document.documentElement.clientHeight + 100) break; - - // otherwise let's add more data + + // let's add more data document.body.insertAdjacentHTML("beforeend", `<p>Date: ${new Date()}</p>`); } } diff --git a/2-ui/3-event-details/8-onscroll/2-updown-button/task.md b/2-ui/3-event-details/8-onscroll/2-updown-button/task.md index 1f648740f..c9f0f6225 100644 --- a/2-ui/3-event-details/8-onscroll/2-updown-button/task.md +++ b/2-ui/3-event-details/8-onscroll/2-updown-button/task.md @@ -11,6 +11,6 @@ It should work like this: - When the page is scrolled down more than the window height -- there appears an "upwards" arrow in the left-top corner. If the page is scrolled back, it disappears. - When the arrow is clicked, the page scrolls to the top. -Like this: +Like this (top-left corner, scroll to see): [iframe border="1" height="200" link src="solution"] diff --git a/2-ui/3-event-details/8-onscroll/3-load-visible-img/solution.md b/2-ui/3-event-details/8-onscroll/3-load-visible-img/solution.md index b07f6820b..1649251b9 100644 --- a/2-ui/3-event-details/8-onscroll/3-load-visible-img/solution.md +++ b/2-ui/3-event-details/8-onscroll/3-load-visible-img/solution.md @@ -1,8 +1,10 @@ The `onscroll` handler should check which images are visible and show them. -We also may want to run it when the page loads, to detect immediately visible images prior to any scrolling and load them. +We also want to run it when the page loads, to detect immediately visible images and load them. -If we put it at the `<body>` bottom, then it runs when the page content is loaded. +The code should execute when the document is loaded, so that it has access to its content. + +Or put it at the `<body>` bottom: ```js // ...the page content is above... @@ -13,12 +15,30 @@ function isVisible(elem) { let windowHeight = document.documentElement.clientHeight; - // top elem edge is visible OR bottom elem edge is visible + // top elem edge is visible? let topVisible = coords.top > 0 && coords.top < windowHeight; + + // bottom elem edge is visible? let bottomVisible = coords.bottom < windowHeight && coords.bottom > 0; return topVisible || bottomVisible; } +``` + +The `showVisible()` function uses the visibility check, implemented by `isVisible()`, to load visible images: + +```js +function showVisible() { + for (let img of document.querySelectorAll('img')) { + let realSrc = img.dataset.src; + if (!realSrc) continue; + + if (isVisible(img)) { + img.src = realSrc; + img.dataset.src = ''; + } + } +} *!* showVisible(); @@ -26,6 +46,4 @@ window.onscroll = showVisible; */!* ``` -For visible images we can take `img.dataset.src` and assign it to `img.src` (if not did it yet). - -P.S. The solution also has a variant of `isVisible` that "pre-loads" images that are within 1 page above/below (the page height is `document.documentElement.clientHeight`). +P.S. The solution also has a variant of `isVisible` that "preloads" images that are within 1 page above/below the current document scroll. diff --git a/2-ui/3-event-details/8-onscroll/3-load-visible-img/source.view/index.html b/2-ui/3-event-details/8-onscroll/3-load-visible-img/source.view/index.html index 5c6027a6f..9953ace66 100644 --- a/2-ui/3-event-details/8-onscroll/3-load-visible-img/source.view/index.html +++ b/2-ui/3-event-details/8-onscroll/3-load-visible-img/source.view/index.html @@ -169,38 +169,8 @@ <h1>Neptune</h1> * It's enough that the top or bottom edge of the element are visible */ function isVisible(elem) { - - let coords = elem.getBoundingClientRect(); - - let windowHeight = document.documentElement.clientHeight; - - // top elem edge is visible OR bottom elem edge is visible - let topVisible = coords.top > 0 && coords.top < windowHeight; - let bottomVisible = coords.bottom < windowHeight && coords.bottom > 0; - - return topVisible || bottomVisible; - } - - /** - A variant of the test that considers the element visible if it's no more than - one page after/behind the current screen. - - function isVisible(elem) { - - let coords = elem.getBoundingClientRect(); - - let windowHeight = document.documentElement.clientHeight; - - let extendedTop = -windowHeight; - let extendedBottom = 2 * windowHeight; - - // top visible || bottom visible - let topVisible = coords.top > extendedTop && coords.top < extendedBottom; - let bottomVisible = coords.bottom < extendedBottom && coords.bottom > extendedTop; - - return topVisible || bottomVisible; + // todo: your code } - */ function showVisible() { for (let img of document.querySelectorAll('img')) { diff --git a/2-ui/3-event-details/8-onscroll/article.md b/2-ui/3-event-details/8-onscroll/article.md index 1eb134f9a..734bd84c6 100644 --- a/2-ui/3-event-details/8-onscroll/article.md +++ b/2-ui/3-event-details/8-onscroll/article.md @@ -1,6 +1,6 @@ # Scrolling -Scroll events allow to react on a page or element scrolling. There are quite a few good things we can do here. +The `scroll` event allows reacting to a page or element scrolling. There are quite a few good things we can do here. For instance: - Show/hide additional controls or information depending on where in the document the user is. @@ -10,7 +10,7 @@ Here's a small function to show the current scroll: ```js autorun window.addEventListener('scroll', function() { - document.getElementById('showScroll').innerHTML = pageYOffset + 'px'; + document.getElementById('showScroll').innerHTML = window.pageYOffset + 'px'; }); ``` @@ -24,14 +24,14 @@ The `scroll` event works both on the `window` and on scrollable elements. ## Prevent scrolling -How do we make something unscrollable? We can't prevent scrolling by using `event.preventDefault()` in `onscroll` listener, because it triggers *after* the scroll has already happened. +How do we make something unscrollable? -But we can prevent scrolling by `event.preventDefault()` on an event that causes the scroll. +We can't prevent scrolling by using `event.preventDefault()` in `onscroll` listener, because it triggers *after* the scroll has already happened. -For instance: -- `wheel` event -- a mouse wheel roll (a "scrolling" touchpad action generates it too). -- `keydown` event for `key:pageUp` and `key:pageDown`. +But we can prevent scrolling by `event.preventDefault()` on an event that causes the scroll, for instance `keydown` event for `key:pageUp` and `key:pageDown`. + +If we add an event handler to these events and `event.preventDefault()` in it, then the scroll won't start. -Sometimes that may help. But there are more ways to scroll, so it's quite hard to handle all of them. So it's more reliable to use CSS to make something unscrollable, like `overflow` property. +There are many ways to initiate a scroll, so it's more reliable to use CSS, `overflow` property. -Here are few tasks that you can solve or look through to see the applications on `onscroll`. +Here are few tasks that you can solve or look through to see applications of `onscroll`. diff --git a/2-ui/4-forms-controls/1-form-elements/1-add-select-option/task.md b/2-ui/4-forms-controls/1-form-elements/1-add-select-option/task.md index 9b218aa7f..a0e74da57 100644 --- a/2-ui/4-forms-controls/1-form-elements/1-add-select-option/task.md +++ b/2-ui/4-forms-controls/1-form-elements/1-add-select-option/task.md @@ -18,3 +18,5 @@ Use JavaScript to: 1. Show the value and the text of the selected option. 2. Add an option: `<option value="classic">Classic</option>`. 3. Make it selected. + +Note, if you've done everything right, your alert should show `blues`. diff --git a/2-ui/4-forms-controls/1-form-elements/article.md b/2-ui/4-forms-controls/1-form-elements/article.md index 5a95d5147..7bc87a0f0 100644 --- a/2-ui/4-forms-controls/1-form-elements/article.md +++ b/2-ui/4-forms-controls/1-form-elements/article.md @@ -2,17 +2,17 @@ Forms and control elements, such as `<input>` have a lot of special properties and events. -Working with forms can be much more convenient if we know them. +Working with forms will be much more convenient when we learn them. ## Navigation: form and elements Document forms are members of the special collection `document.forms`. -That's a *named* collection: we can use both the name and the number to get the form. +That's a so-called *"named collection"*: it's both named and ordered. We can use both the name or the number in the document to get the form. ```js no-beautify -document.forms.my - the form with name="my" -document.forms[0] - the first form in the document +document.forms.my; // the form with name="my" +document.forms[0]; // the first form in the document ``` When we have a form, then any element is available in the named collection `form.elements`. @@ -36,9 +36,9 @@ For instance: </script> ``` -There may be multiple elements with the same name, that's often the case with radio buttons. +There may be multiple elements with the same name. This is typical with radio buttons and checkboxes. -In that case `form.elements[name]` is a collection, for instance: +In that case, `form.elements[name]` is a *collection*. For instance: ```html run height=40 <form> @@ -51,15 +51,17 @@ let form = document.forms[0]; let ageElems = form.elements.age; -alert(ageElems[0].value); // 10, the first input value +*!* +alert(ageElems[0]); // [object HTMLInputElement] +*/!* </script> ``` -These navigation properties do not depend on the tag structure. All elements, no matter how deep they are in the form, are available in `form.elements`. +These navigation properties do not depend on the tag structure. All control elements, no matter how deep they are in the form, are available in `form.elements`. ````smart header="Fieldsets as \"subforms\"" -A form may have one or many `<fieldset>` elements inside it. They also support the `elements` property. +A form may have one or many `<fieldset>` elements inside it. They also have `elements` property that lists form controls inside them. For instance: @@ -79,7 +81,7 @@ For instance: let fieldset = form.elements.userFields; alert(fieldset); // HTMLFieldSetElement - // we can get the input both from the form and from the fieldset + // we can get the input by name both from the form and from the fieldset alert(fieldset.elements.login == form.elements.login); // true */!* </script> @@ -90,7 +92,7 @@ For instance: ````warn header="Shorter notation: `form.name`" There's a shorter notation: we can access the element as `form[index/name]`. -Instead of `form.elements.login` we can write `form.login`. +In other words, instead of `form.elements.login` we can write `form.login`. That also works, but there's a minor issue: if we access an element, and then change its `name`, then it is still available under the old name (as well as under the new one). @@ -111,20 +113,19 @@ That's easy to see in an example: alert(form.elements.username); // input *!* - // the direct access now can use both names: the new one and the old one + // form allows both names: the new one and the old one alert(form.username == form.login); // true */!* </script> ``` -That's usually not a problem, because we rarely change names of form elements. +That's usually not a problem, however, because we rarely change names of form elements. ```` ## Backreference: element.form -For any element, the form is available as `element.form`. So a form references all elements, and elements -reference the form. +For any element, the form is available as `element.form`. So a form references all elements, and elements reference the form. Here's the picture: @@ -150,11 +151,11 @@ For instance: ## Form elements -Let's talk about form controls, pay attention to their specific features. +Let's talk about form controls. ### input and textarea -Normally, we can access the value as `input.value` or `input.checked` for checkboxes. +We can access their value as `input.value` (string) or `input.checked` (boolean) for checkboxes and radio buttons. Like this: @@ -166,26 +167,26 @@ input.checked = true; // for a checkbox or radio button ``` ```warn header="Use `textarea.value`, not `textarea.innerHTML`" -Please note that we should never use `textarea.innerHTML`: it stores only the HTML that was initially on the page, not the current value. +Please note that even though `<textarea>...</textarea>` holds its value as nested HTML, we should never use `textarea.innerHTML` to access it. + +It stores only the HTML that was initially on the page, not the current value. ``` ### select and option A `<select>` element has 3 important properties: -1. `select.options` -- the collection of `<option>` elements, -2. `select.value` -- the value of the chosen option, -3. `select.selectedIndex` -- the number of the selected option. - -So we have three ways to set the value of a `<select>`: +1. `select.options` -- the collection of `<option>` subelements, +2. `select.value` -- the *value* of the currently selected `<option>`, +3. `select.selectedIndex` -- the *number* of the currently selected `<option>`. -1. Find the needed `<option>` and set `option.selected` to `true`. -2. Set `select.value` to the value. -3. Set `select.selectedIndex` to the number of the option. +They provide three different ways of setting a value for a `<select>`: -The first way is the most obvious, but `(2)` and `(3)` are usually more convenient. +1. Find the corresponding `<option>` element (e.g. among `select.options`) and set its `option.selected` to `true`. +2. If we know a new value: set `select.value` to the new value. +3. If we know the new option number: set `select.selectedIndex` to that number. -Here is an example: +Here is an example of all three methods: ```html run <select id="select"> @@ -196,15 +197,18 @@ Here is an example: <script> // all three lines do the same thing - select.options[2].selected = true; + select.options[2].selected = true; select.selectedIndex = 2; select.value = 'banana'; + // please note: options start from zero, so index 2 means the 3rd option. </script> ``` -Unlike most other controls, `<select multiple>` allows multiple choice. In that case we need to walk over `select.options` to get all selected values. +Unlike most other controls, `<select>` allows to select multiple options at once if it has `multiple` attribute. This attribute is rarely used, though. -Like this: +For multiple selected values, use the first way of setting values: add/remove the `selected` property from `<option>` subelements. + +Here's an example of how to get selected values from a multi-select: ```html run <select id="select" *!*multiple*/!*> @@ -223,48 +227,54 @@ Like this: </script> ``` -The full specification of the `<select>` element is available at <https://html.spec.whatwg.org/multipage/forms.html#the-select-element>. +The full specification of the `<select>` element is available in the specification <https://html.spec.whatwg.org/multipage/forms.html#the-select-element>. ### new Option -In the specification of [the option element](https://html.spec.whatwg.org/multipage/forms.html#the-option-element) there's a nice short syntax to create `<option>` elements: +In the [specification](https://html.spec.whatwg.org/multipage/forms.html#the-option-element) there's a nice short syntax to create an `<option>` element: ```js option = new Option(text, value, defaultSelected, selected); ``` -Parameters: +This syntax is optional. We can use `document.createElement('option')` and set attributes manually. Still, it may be shorter, so here are the parameters: - `text` -- the text inside the option, - `value` -- the option value, -- `defaultSelected` -- if `true`, then `selected` attribute is created, +- `defaultSelected` -- if `true`, then `selected` HTML-attribute is created, - `selected` -- if `true`, then the option is selected. -For instance: +The difference between `defaultSelected` and `selected` is that `defaultSelected` sets the HTML-attribute (that we can get using `option.getAttribute('selected')`), while `selected` sets whether the option is selected or not. + +In practice, one should usually set _both_ values to `true` or `false`. (Or, simply omit them; both default to `false`.) + +For instance, here's a new "unselected" option: ```js let option = new Option("Text", "value"); // creates <option value="value">Text</option> ``` -The same element selected: +The same option, but selected: ```js let option = new Option("Text", "value", true, true); ``` -```smart header="Additional properties of `<option>`" -Option elements have additional properties: +Option elements have properties: -`selected` +`option.selected` : Is the option selected. -`index` +`option.index` : The number of the option among the others in its `<select>`. -`text` +`option.text` : Text content of the option (seen by the visitor). -``` + +## References + +- Specification: <https://html.spec.whatwg.org/multipage/forms.html>. ## Summary @@ -279,8 +289,10 @@ Form navigation: `element.form` : Elements reference their form in the `form` property. -Value is available as `input.value`, `textarea.value`, `select.value` etc, or `input.checked` for checkboxes and radio buttons. +Value is available as `input.value`, `textarea.value`, `select.value`, etc. (For checkboxes and radio buttons, use `input.checked` to determine whether a value is selected.) + +For `<select>`, one can also get the value by the index `select.selectedIndex` or through the options collection `select.options`. -For `<select>` we can also get the value by the index `select.selectedIndex` or through the options collection `select.options`. The full specification of this and other elements is at <https://html.spec.whatwg.org/multipage/forms.html>. +These are the basics to start working with forms. We'll meet many examples further in the tutorial. -These are the basics to start working with forms. In the next chapter we'll cover `focus` and `blur` events that may occur on any element, but are mostly handled on forms. +In the next chapter we'll cover `focus` and `blur` events that may occur on any element, but are mostly handled on forms. diff --git a/2-ui/4-forms-controls/2-focus-blur/3-editable-div/solution.view/my.css b/2-ui/4-forms-controls/2-focus-blur/3-editable-div/solution.view/my.css index 67905e6d6..bf3d4ff6a 100644 --- a/2-ui/4-forms-controls/2-focus-blur/3-editable-div/solution.view/my.css +++ b/2-ui/4-forms-controls/2-focus-blur/3-editable-div/solution.view/my.css @@ -20,6 +20,6 @@ } .edit:focus { - outline: none; /* remove focus border in Safari */ + outline: none; } diff --git a/2-ui/4-forms-controls/2-focus-blur/3-editable-div/source.view/my.css b/2-ui/4-forms-controls/2-focus-blur/3-editable-div/source.view/my.css index 67905e6d6..bf3d4ff6a 100644 --- a/2-ui/4-forms-controls/2-focus-blur/3-editable-div/source.view/my.css +++ b/2-ui/4-forms-controls/2-focus-blur/3-editable-div/source.view/my.css @@ -20,6 +20,6 @@ } .edit:focus { - outline: none; /* remove focus border in Safari */ + outline: none; } diff --git a/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/solution.view/my.css b/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/solution.view/my.css index d7f2a4424..8d0c3b4ec 100644 --- a/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/solution.view/my.css +++ b/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/solution.view/my.css @@ -3,14 +3,15 @@ margin: 0; padding: 0; display: block; - resize: none; + /* remove resizing handle in Firefox */ + resize: none; - outline: none; /* remove outline on focus in Chrome */ + outline: none; - overflow: auto; /* remove scrollbar in IE */ + overflow: auto; } .edit-controls { diff --git a/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/task.md b/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/task.md index 2cccea020..378bd1f54 100644 --- a/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/task.md +++ b/2-ui/4-forms-controls/2-focus-blur/4-edit-td-click/task.md @@ -6,7 +6,7 @@ importance: 5 Make table cells editable on click. -- On click -- the cell should became "editable" (textarea appears inside), we can change HTML. There should be no resize, all geometry should remain the same. +- On click -- the cell should become "editable" (textarea appears inside), we can change HTML. There should be no resize, all geometry should remain the same. - Buttons OK and CANCEL appear below the cell to finish/cancel the editing. - Only one cell may be editable at a moment. While a `<td>` is in "edit mode", clicks on other cells are ignored. - The table may have many cells. Use event delegation. diff --git a/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/solution.md b/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/solution.md index 2c5c73cd3..4d1682176 100644 --- a/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/solution.md +++ b/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/solution.md @@ -1,5 +1,5 @@ -We can use `mouse.onclick` to handle the click and make the mouse "moveable" with `position:fixed`, then then `mouse.onkeydown` to handle arrow keys. +We can use `mouse.onclick` to handle the click and make the mouse "moveable" with `position:fixed`, then `mouse.onkeydown` to handle arrow keys. The only pitfall is that `keydown` only triggers on elements with focus. So we need to add `tabindex` to the element. As we're forbidden to change HTML, we can use `mouse.tabIndex` property for that. diff --git a/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/task.md b/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/task.md index fc48c21ff..644d814d9 100644 --- a/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/task.md +++ b/2-ui/4-forms-controls/2-focus-blur/5-keyboard-mouse/task.md @@ -9,4 +9,5 @@ Focus on the mouse. Then use arrow keys to move it: [demo src="solution"] P.S. Don't put event handlers anywhere except the `#mouse` element. + P.P.S. Don't modify HTML/CSS, the approach should be generic and work with any element. diff --git a/2-ui/4-forms-controls/2-focus-blur/article.md b/2-ui/4-forms-controls/2-focus-blur/article.md index 1a880ce47..c253dc11d 100644 --- a/2-ui/4-forms-controls/2-focus-blur/article.md +++ b/2-ui/4-forms-controls/2-focus-blur/article.md @@ -1,14 +1,14 @@ # Focusing: focus/blur -An element receives a focus when the user either clicks on it or uses the `key:Tab` key on the keyboard. There's also an `autofocus` HTML attribute that puts the focus into an element by default when a page loads and other means of getting a focus. +An element receives the focus when the user either clicks on it or uses the `key:Tab` key on the keyboard. There's also an `autofocus` HTML attribute that puts the focus onto an element by default when a page loads and other means of getting the focus. -Focusing generally means: "prepare to accept the data here", so that's the moment when we can run the code to initialize or load something. +Focusing on an element generally means: "prepare to accept the data here", so that's the moment when we can run the code to initialize the required functionality. The moment of losing the focus ("blur") can be even more important. That's when a user clicks somewhere else or presses `key:Tab` to go to the next form field, or there are other means as well. Losing the focus generally means: "the data has been entered", so we can run the code to check it or even to save it to the server and so on. -There are important peculiarities when working with focus events. We'll do the best to cover them here. +There are important peculiarities when working with focus events. We'll do the best to cover them further on. ## Events focus/blur @@ -18,7 +18,7 @@ Let's use them for validation of an input field. In the example below: -- The `blur` handler checks if the field the email is entered, and if not -- shows an error. +- The `blur` handler checks if the field has an email entered, and if not -- shows an error. - The `focus` handler hides the error message (on `blur` it will be checked again): ```html run autorun height=60 @@ -49,7 +49,7 @@ Your email please: <input type="email" id="input"> </script> ``` -Modern HTML allows to do many validations using input attributes: `required`, `pattern` and so on. And sometimes they are just what we need. JavaScript can be used when we want more flexibility. Also we could automatically send the changed value to the server if it's correct. +Modern HTML allows us to do many validations using input attributes: `required`, `pattern` and so on. And sometimes they are just what we need. JavaScript can be used when we want more flexibility. Also we could automatically send the changed value to the server if it's correct. ## Methods focus/blur @@ -90,6 +90,8 @@ If we enter something into the input and then try to use `key:Tab` or click away Please note that we can't "prevent losing focus" by calling `event.preventDefault()` in `onblur`, because `onblur` works *after* the element lost the focus. +In practice though, one should think well, before implementing something like this, because we generally *should show errors* to the user, but *should not prevent their progress* in filling our form. They may want to fill other fields first. + ```warn header="JavaScript-initiated focus loss" A focus loss can occur for many reasons. @@ -104,29 +106,34 @@ The best recipe is to be careful when using these events. If we want to track us ``` ## Allow focusing on any element: tabindex -By default many elements do not support focusing. +By default, many elements do not support focusing. -The list varies between browsers, but one thing is always correct: `focus/blur` support is guaranteed for elements that a visitor can interact with: `<button>`, `<input>`, `<select>`, `<a>` and so on. +The list varies a bit between browsers, but one thing is always correct: `focus/blur` support is guaranteed for elements that a visitor can interact with: `<button>`, `<input>`, `<select>`, `<a>` and so on. -From the other hand, elements that exist to format something like `<div>`, `<span>`, `<table>` -- are unfocusable by default. The method `elem.focus()` doesn't work on them, and `focus/blur` events are never triggered. +On the other hand, elements that exist to format something, such as `<div>`, `<span>`, `<table>` -- are unfocusable by default. The method `elem.focus()` doesn't work on them, and `focus/blur` events are never triggered. This can be changed using HTML-attribute `tabindex`. -The purpose of this attribute is to specify the order number of the element when `key:Tab` is used to switch between them. +Any element becomes focusable if it has `tabindex`. The value of the attribute is the order number of the element when `key:Tab` (or something like that) is used to switch between them. + +That is: if we have two elements, the first has `tabindex="1"`, and the second has `tabindex="2"`, then pressing `key:Tab` while in the first element -- moves the focus into the second one. + +The switch order is: elements with `tabindex` from `1` and above go first (in the `tabindex` order), and then elements without `tabindex` (e.g. a regular `<input>`). -That is: if we have two elements, the first has `tabindex="1"`, and the second has `tabindex="2"`, then pressing `key:Tab` while in the first element -- moves us to the second one. +Elements without matching `tabindex` are switched in the document source order (the default order). There are two special values: -- `tabindex="0"` makes the element the last one. -- `tabindex="-1"` means that `key:Tab` should ignore that element. +- `tabindex="0"` puts an element among those without `tabindex`. That is, when we switch elements, elements with `tabindex=0` go after elements with `tabindex ≥ 1`. -**Any element supports focusing if it has `tabindex`.** + Usually it's used to make an element focusable, but keep the default switching order. To make an element a part of the form on par with `<input>`. + +- `tabindex="-1"` allows only programmatic focusing on an element. The `key:Tab` key ignores such elements, but method `elem.focus()` works. For instance, here's a list. Click the first item and press `key:Tab`: ```html autorun no-beautify -Click the first item and press Tab. Keep track of the order. Please note that many subsequent Tabs can move the focus out of the iframe with the example. +Click the first item and press Tab. Keep track of the order. Please note that many subsequent Tabs can move the focus out of the iframe in the example. <ul> <li tabindex="1">One</li> <li tabindex="0">Zero</li> @@ -140,9 +147,9 @@ Click the first item and press Tab. Keep track of the order. Please note that ma </style> ``` -The order is like this: `1 - 2 - 0` (zero is always the last). Normally, `<li>` does not support focusing, but `tabindex` full enables it, along with events and styling with `:focus`. +The order is like this: `1 - 2 - 0`. Normally, `<li>` does not support focusing, but `tabindex` full enables it, along with events and styling with `:focus`. -```smart header="`elem.tabIndex` works too" +```smart header="The property `elem.tabIndex` works too" We can add `tabindex` from JavaScript by using the `elem.tabIndex` property. That has the same effect. ``` @@ -203,7 +210,6 @@ So here's another working variant: <script> *!* - // put the handler on capturing phase (last argument true) form.addEventListener("focusin", () => form.classList.add('focused')); form.addEventListener("focusout", () => form.classList.remove('focused')); */!* @@ -212,7 +218,7 @@ So here's another working variant: ## Summary -Events `focus` and `blur` trigger on focusing/losing focus on the element. +Events `focus` and `blur` trigger on an element focusing/losing focus. Their specials are: - They do not bubble. Can use capturing state instead or `focusin/focusout`. diff --git a/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/solution.view/index.html b/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/solution.view/index.html index 4850b2ca9..0515c839e 100644 --- a/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/solution.view/index.html +++ b/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/solution.view/index.html @@ -96,7 +96,7 @@ let years = form.months.value / 12; if (!years) return; - let result = Math.round(initial * (1 + interest * years)); + let result = Math.round(initial * (1 + interest) ** years); let height = result / form.money.value * 100 + 'px'; document.getElementById('height-after').style.height = height; diff --git a/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/task.md b/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/task.md index e324577a9..73f0477ff 100644 --- a/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/task.md +++ b/2-ui/4-forms-controls/3-events-change-input/1-deposit-calculator/task.md @@ -17,5 +17,5 @@ The formula is: // initial: the initial money sum // interest: e.g. 0.05 means 5% per year // years: how many years to wait -let result = Math.round(initial * (1 + interest * years)); +let result = Math.round(initial * (1 + interest) ** years); ``` diff --git a/2-ui/4-forms-controls/3-events-change-input/article.md b/2-ui/4-forms-controls/3-events-change-input/article.md index b6247198d..480197ae5 100644 --- a/2-ui/4-forms-controls/3-events-change-input/article.md +++ b/2-ui/4-forms-controls/3-events-change-input/article.md @@ -1,10 +1,10 @@ # Events: change, input, cut, copy, paste -Let's discuss various events that accompany data updates. +Let's cover various events that accompany data updates. ## Event: change -The [change](http://www.w3.org/TR/html5/forms.html#event-input-change) event triggers when the element has finished changing. +The `change` event triggers when the element has finished changing. For text inputs that means that the event occurs when it loses focus. @@ -15,11 +15,23 @@ For instance, while we are typing in the text field below -- there's no event. B <input type="button" value="Button"> ``` -For other elements: `select`, `input type=checkbox/radio` it triggers right after the selection changes. +For other elements: `select`, `input type=checkbox/radio` it triggers right after the selection changes: + +```html autorun height=40 run +<select onchange="alert(this.value)"> + <option value="">Select something</option> + <option value="1">Option 1</option> + <option value="2">Option 2</option> + <option value="3">Option 3</option> +</select> +``` + ## Event: input -The `input` event triggers every time a value is modified. +The `input` event triggers every time after a value is modified by the user. + +Unlike keyboard events, it triggers on any value change, even those that does not involve keyboard actions: pasting with a mouse or using speech recognition to dictate the text. For instance: @@ -34,7 +46,7 @@ For instance: If we want to handle every modification of an `<input>` then this event is the best choice. -Unlike keyboard events it works on any value change, even those that does not involve keyboard actions: pasting with a mouse or using speech recognition to dictate the text. +On the other hand, `input` event doesn't trigger on keyboard input and other actions that do not involve value change, e.g. pressing arrow keys `key:⇦` `key:⇨` while in the input. ```smart header="Can't prevent anything in `oninput`" The `input` event occurs after the value is modified. @@ -46,27 +58,50 @@ So we can't use `event.preventDefault()` there -- it's just too late, there woul These events occur on cutting/copying/pasting a value. -They belong to [ClipboardEvent](https://www.w3.org/TR/clipboard-apis/#clipboard-event-interfaces) class and provide access to the data that is copied/pasted. +They belong to [ClipboardEvent](https://www.w3.org/TR/clipboard-apis/#clipboard-event-interfaces) class and provide access to the data that is cut/copied/pasted. -We also can use `event.preventDefault()` to abort the action. +We also can use `event.preventDefault()` to abort the action, then nothing gets copied/pasted. -For instance, the code below prevents all such events and shows what we are trying to cut/copy/paste: +For instance, the code below prevents all `cut/copy/paste` events and shows the text we're trying to cut/copy/paste: ```html autorun height=40 run <input type="text" id="input"> <script> - input.oncut = input.oncopy = input.onpaste = function(event) { - alert(event.type + ' - ' + event.clipboardData.getData('text/plain')); - return false; + input.onpaste = function(event) { + alert("paste: " + event.clipboardData.getData('text/plain')); + event.preventDefault(); + }; + + input.oncut = input.oncopy = function(event) { + alert(event.type + '-' + document.getSelection()); + event.preventDefault(); }; </script> ``` -Technically, we can copy/paste everything. For instance, we can copy a file in the OS file manager, and paste it. +Please note: inside `cut` and `copy` event handlers a call to `event.clipboardData.getData(...)` returns an empty string. That's because technically the data isn't in the clipboard yet. If we use `event.preventDefault()` it won't be copied at all. + +So the example above uses `document.getSelection()` to get the selected text. You can find more details about document selection in the article <info:selection-range>. + +It's possible to copy/paste not just text, but everything. For instance, we can copy a file in the OS file manager, and paste it. + +That's because `clipboardData` implements `DataTransfer` interface, commonly used for drag'n'drop and copy/pasting. It's a bit beyond our scope now, but you can find its methods in the [DataTransfer specification](https://html.spec.whatwg.org/multipage/dnd.html#the-datatransfer-interface). + +Also, there's an additional asynchronous API of accessing the clipboard: `navigator.clipboard`. More about it in the specification [Clipboard API and events](https://www.w3.org/TR/clipboard-apis/), [not supported by Firefox](https://caniuse.com/async-clipboard). + +### Safety restrictions + +The clipboard is a "global" OS-level thing. A user may switch between various applications, copy/paste different things, and a browser page shouldn't see all that. + +So most browsers allow seamless read/write access to the clipboard only in the scope of certain user actions, such as copying/pasting etc. + +It's forbidden to generate "custom" clipboard events with `dispatchEvent` in all browsers except Firefox. And even if we manage to dispatch such event, the specification clearly states that such "synthetic" events must not provide access to the clipboard. + +Even if someone decides to save `event.clipboardData` in an event handler, and then access it later -- it won't work. -There's a list of methods [in the specification](https://www.w3.org/TR/clipboard-apis/#dfn-datatransfer) to work with different data types, read/write to the clipboard. +To reiterate, [event.clipboardData](https://www.w3.org/TR/clipboard-apis/#clipboardevent-clipboarddata) works solely in the context of user-initiated event handlers. -But please note that clipboard is a "global" OS-level thing. Most browsers allow read/write access to the clipboard only in the scope of certain user actions for the safety. Also it is forbidden to create "custom" clipboard events in all browsers except Firefox. +On the other hand, [navigator.clipboard](https://www.w3.org/TR/clipboard-apis/#h-navigator-clipboard) is the more recent API, meant for use in any context. It asks for user permission, if needed. ## Summary @@ -76,4 +111,4 @@ Data change events: |---------|----------|-------------| | `change`| A value was changed. | For text inputs triggers on focus loss. | | `input` | For text inputs on every change. | Triggers immediately unlike `change`. | -| `cut/copy/paste` | Cut/copy/paste actions. | The action can be prevented. The `event.clipboardData` property gives read/write access to the clipboard. | +| `cut/copy/paste` | Cut/copy/paste actions. | The action can be prevented. The `event.clipboardData` property gives access to the clipboard. All browsers except Firefox also support `navigator.clipboard`. | diff --git a/2-ui/4-forms-controls/4-forms-submit/article.md b/2-ui/4-forms-controls/4-forms-submit/article.md index 69d5a3dc2..c00c559c0 100644 --- a/2-ui/4-forms-controls/4-forms-submit/article.md +++ b/2-ui/4-forms-controls/4-forms-submit/article.md @@ -1,4 +1,4 @@ -# Form submission: event and method submit +# Forms: event and method submit The `submit` event triggers when the form is submitted, it is usually used to validate the form before sending it to the server or to abort the submission and process it in JavaScript. diff --git a/2-ui/5-loading/01-onload-ondomcontentloaded/article.md b/2-ui/5-loading/01-onload-ondomcontentloaded/article.md index 2d60aa1d3..07624a658 100644 --- a/2-ui/5-loading/01-onload-ondomcontentloaded/article.md +++ b/2-ui/5-loading/01-onload-ondomcontentloaded/article.md @@ -2,14 +2,14 @@ The lifecycle of an HTML page has three important events: -- `DOMContentLoaded` -- the browser fully loaded HTML, and the DOM tree is built, but external resources like pictures `<img>` and stylesheets may be not yet loaded. -- `load` -- the browser loaded all resources (images, styles etc). -- `beforeunload/unload` -- when the user is leaving the page. +- `DOMContentLoaded` -- the browser fully loaded HTML, and the DOM tree is built, but external resources like pictures `<img>` and stylesheets may not yet have loaded. +- `load` -- not only HTML is loaded, but also all the external resources: images, styles etc. +- `beforeunload/unload` -- the user is leaving the page. Each event may be useful: - `DOMContentLoaded` event -- DOM is ready, so the handler can lookup DOM nodes, initialize the interface. -- `load` event -- additional resources are loaded, we can get image sizes (if not specified in HTML/CSS) etc. +- `load` event -- external resources are loaded, so styles are applied, image sizes are known etc. - `beforeunload` event -- the user is leaving: we can check if the user saved the changes and ask them whether they really want to leave. - `unload` -- the user almost left, but we still can initiate some operations, such as sending out statistics. @@ -33,7 +33,7 @@ For instance: function ready() { alert('DOM is ready'); - // image is not yet loaded (unless was cached), so the size is 0x0 + // image is not yet loaded (unless it was cached), so the size is 0x0 alert(`Image size: ${img.offsetWidth}x${img.offsetHeight}`); } @@ -45,11 +45,11 @@ For instance: <img id="img" src="https://en.js.cx/clipart/train.gif?speed=1&cache=0"> ``` -In the example the `DOMContentLoaded` handler runs when the document is loaded, so it can see all the elements, including `<img>` below. +In the example, the `DOMContentLoaded` handler runs when the document is loaded, so it can see all the elements, including `<img>` below. But it doesn't wait for the image to load. So `alert` shows zero sizes. -At the first sight `DOMContentLoaded` event is very simple. The DOM tree is ready -- here's the event. There are few peculiarities though. +At first sight, the `DOMContentLoaded` event is very simple. The DOM tree is ready -- here's the event. There are few peculiarities though. ### DOMContentLoaded and scripts @@ -60,7 +60,7 @@ So DOMContentLoaded definitely happens after such scripts: ```html run <script> document.addEventListener("DOMContentLoaded", () => { - console.log("DOM ready!"); + alert("DOM ready!"); }); </script> @@ -73,11 +73,10 @@ So DOMContentLoaded definitely happens after such scripts: In the example above, we first see "Library loaded...", and then "DOM ready!" (all scripts are executed). -```warn header="Scripts with `async`, `defer` or `type=\"module\"` don't block DOMContentLoaded" - -Script attributes `async` and `defer`, that we'll cover [a bit later](info:script-async-defer), don't block DOMContentLoaded. [JavaScript modules](info:modules) behave like `defer`, they don't block it too. - -So here we're talking about "regular" scripts, like `<script>...</script>`, or `<script src="..."></script>`. +```warn header="Scripts that don't block DOMContentLoaded" +There are two exceptions from this rule: +1. Scripts with the `async` attribute, that we'll cover [a bit later](info:script-async-defer), don't block `DOMContentLoaded`. +2. Scripts that are generated dynamically with `document.createElement('script')` and then added to the webpage also don't block this event. ``` ### DOMContentLoaded and styles @@ -86,15 +85,15 @@ External style sheets don't affect DOM, so `DOMContentLoaded` does not wait for But there's a pitfall. If we have a script after the style, then that script must wait until the stylesheet loads: -```html +```html run <link type="text/css" rel="stylesheet" href="style.css"> <script> - // the script doesn't not execute until the stylesheet is loaded + // the script doesn't execute until the stylesheet is loaded alert(getComputedStyle(document.body).marginTop); </script> ``` -The reason is that the script may want to get coordinates and other style-dependent properties of elements, like in the example above. Naturally, it has to wait for styles to load. +The reason for this is that the script may want to get coordinates and other style-dependent properties of elements, like in the example above. Naturally, it has to wait for styles to load. As `DOMContentLoaded` waits for scripts, it now waits for styles before them as well. @@ -109,13 +108,13 @@ So if `DOMContentLoaded` is postponed by long-loading scripts, then autofill als ## window.onload [#window-onload] -The `load` event on the `window` object triggers when the whole page is loaded including styles, images and other resources. +The `load` event on the `window` object triggers when the whole page is loaded including styles, images and other resources. This event is available via the `onload` property. The example below correctly shows image sizes, because `window.onload` waits for all images: ```html run height=200 refresh <script> - window.onload = function() { + window.onload = function() { // can also use window.addEventListener('load', (event) => { alert('Page loaded'); // image is loaded at this time @@ -146,19 +145,19 @@ let analyticsData = { /* object with gathered data */ }; window.addEventListener("unload", function() { navigator.sendBeacon("/analytics", JSON.stringify(analyticsData)); -}; +}); ``` - The request is sent as POST. -- We can send not only a string, but also forms and other formats, as described in the chapter <info:fetch-basics>, but usually it's a stringified object. +- We can send not only a string, but also forms and other formats, as described in the chapter <info:fetch>, but usually it's a stringified object. - The data is limited by 64kb. When the `sendBeacon` request is finished, the browser probably has already left the document, so there's no way to get server response (which is usually empty for analytics). -There's also a `keepalive` flag for doing such "after-page-left" requests in [fetch](info:fetch-basics) method for generic network requests. You can find more information in the chapter <info:fetch-api>. +There's also a `keepalive` flag for doing such "after-page-left" requests in [fetch](info:fetch) method for generic network requests. You can find more information in the chapter <info:fetch-api>. -If we want to cancel the transition to another page, we can't do it here. But we can use another event -- `onbeforeunload`. +If we want to cancel the transition to another page, we can't do it here. But we can use another event -- `onbeforeunload`. ## window.onbeforeunload [#window.onbeforeunload] @@ -174,7 +173,7 @@ window.onbeforeunload = function() { }; ``` -For historical reasons, returning a non-empty string also counts as canceling the event. Some time ago browsers used show it as a message, but as the [modern specification](https://html.spec.whatwg.org/#unloading-documents) says, they shouldn't. +For historical reasons, returning a non-empty string also counts as canceling the event. Some time ago browsers used to show it as a message, but as the [modern specification](https://html.spec.whatwg.org/#unloading-documents) says, they shouldn't. Here's an example: @@ -186,6 +185,26 @@ window.onbeforeunload = function() { The behavior was changed, because some webmasters abused this event handler by showing misleading and annoying messages. So right now old browsers still may show it as a message, but aside of that -- there's no way to customize the message shown to the user. +````warn header="The `event.preventDefault()` doesn't work from a `beforeunload` handler" +That may sound weird, but most browsers ignore `event.preventDefault()`. + +Which means, following code may not work: +```js run +window.addEventListener("beforeunload", (event) => { + // doesn't work, so this event handler doesn't do anything + event.preventDefault(); +}); +``` + +Instead, in such handlers one should set `event.returnValue` to a string to get the result similar to the code above: +```js run +window.addEventListener("beforeunload", (event) => { + // works, same as returning from window.onbeforeunload + event.returnValue = "There are unsaved changes. Leave now?"; +}); +``` +```` + ## readyState What happens if we set the `DOMContentLoaded` handler after the document is loaded? @@ -210,7 +229,7 @@ Like this: function work() { /*...*/ } if (document.readyState == 'loading') { - // loading yet, wait for the event + // still loading, wait for the event document.addEventListener('DOMContentLoaded', work); } else { // DOM is ready! @@ -218,7 +237,7 @@ if (document.readyState == 'loading') { } ``` -There's also `readystatechange` event that triggers when the state changes, so we can print all these states like this: +There's also the `readystatechange` event that triggers when the state changes, so we can print all these states like this: ```js run // current state @@ -246,7 +265,7 @@ Here's a document with `<iframe>`, `<img>` and handlers that log events: <iframe src="iframe.html" onload="log('iframe onload')"></iframe> -<img src="http://en.js.cx/clipart/train.gif" id="img"> +<img src="https://en.js.cx/clipart/train.gif" id="img"> <script> img.onload = () => log('img onload'); </script> @@ -273,12 +292,12 @@ The numbers in square brackets denote the approximate time of when it happens. E Page load events: -- `DOMContentLoaded` event triggers on `document` when DOM is ready. We can apply JavaScript to elements at this stage. +- The `DOMContentLoaded` event triggers on `document` when the DOM is ready. We can apply JavaScript to elements at this stage. - Script such as `<script>...</script>` or `<script src="..."></script>` block DOMContentLoaded, the browser waits for them to execute. - Images and other resources may also still continue loading. -- `load` event on `window` triggers when the page and all resources are loaded. We rarely use it, because there's usually no need to wait for so long. -- `beforeunload` event on `window` triggers when the user wants to leave the page. If we cancel the event, browser asks whether the user really wants to leave (e.g we have unsaved changes). -- `unload` event on `window` triggers when the user is finally leaving, in the handler we can only do simple things that do not involve delays or asking a user. Because of that limitation, it's rarely used. We can send out a network request with `navigator.sendBeacon`. +- The `load` event on `window` triggers when the page and all resources are loaded. We rarely use it, because there's usually no need to wait for so long. +- The `beforeunload` event on `window` triggers when the user wants to leave the page. If we cancel the event, browser asks whether the user really wants to leave (e.g we have unsaved changes). +- The `unload` event on `window` triggers when the user is finally leaving, in the handler we can only do simple things that do not involve delays or asking a user. Because of that limitation, it's rarely used. We can send out a network request with `navigator.sendBeacon`. - `document.readyState` is the current state of the document, changes can be tracked in the `readystatechange` event: - `loading` -- the document is loading. - `interactive` -- the document is parsed, happens at about the same time as `DOMContentLoaded`, but before it. diff --git a/2-ui/5-loading/01-onload-ondomcontentloaded/readystate.view/index.html b/2-ui/5-loading/01-onload-ondomcontentloaded/readystate.view/index.html index a4685a716..27df70939 100644 --- a/2-ui/5-loading/01-onload-ondomcontentloaded/readystate.view/index.html +++ b/2-ui/5-loading/01-onload-ondomcontentloaded/readystate.view/index.html @@ -9,8 +9,8 @@ [20] readyState:interactive [21] DOMContentLoaded [30] iframe onload - [40] readyState:complete [40] img onload + [40] readyState:complete [40] window onload --> diff --git a/2-ui/5-loading/02-script-async-defer/article.md b/2-ui/5-loading/02-script-async-defer/article.md index b5fab5c06..f97c000d6 100644 --- a/2-ui/5-loading/02-script-async-defer/article.md +++ b/2-ui/5-loading/02-script-async-defer/article.md @@ -3,11 +3,11 @@ In modern websites, scripts are often "heavier" than HTML: their download size is larger, and processing time is also longer. -When the browser loads HTML and comes across a `<script>...</script>` tag, it can't continue building DOM. It must execute the script right now. The same happens for external scripts `<script src="..."></script>`: the browser must wait until the script downloads, execute it, and only after process the rest of the page. +When the browser loads HTML and comes across a `<script>...</script>` tag, it can't continue building the DOM. It must execute the script right now. The same happens for external scripts `<script src="..."></script>`: the browser must wait for the script to download, execute the downloaded script, and only then can it process the rest of the page. That leads to two important issues: -1. Scripts can't see DOM elements below them, so can't add handlers etc. +1. Scripts can't see DOM elements below them, so they can't add handlers etc. 2. If there's a bulky script at the top of the page, it "blocks the page". Users can't see the page content till it downloads and runs: ```html run height=100 @@ -29,15 +29,15 @@ There are some workarounds to that. For instance, we can put a script at the bot </body> ``` -But this solution is far from perfect. For example, the browser actually notices the script (and can start downloading it) only after it downloaded the full HTML document. For long HTML documents, that may be a noticeable delay. +But this solution is far from perfect. For example, the browser notices the script (and can start downloading it) only after it downloaded the full HTML document. For long HTML documents, that may be a noticeable delay. -Such things are invisible for people using very fast connections, but many people in the world still have slower internet speeds and far-from-perfect mobile connectivity. +Such things are invisible for people using very fast connections, but many people in the world still have slow internet speeds and use a far-from-perfect mobile internet connection. -Luckily, there are two `<script>` attributes that solve the problem for us: `defer` and `async` +Luckily, there are two `<script>` attributes that solve the problem for us: `defer` and `async`. ## defer -The `defer` attribute tells the browser that it should go on working with the page, and load the script "in background", then run the script when it loads. +The `defer` attribute tells the browser not to wait for the script. Instead, the browser will continue to process the HTML, build DOM. The script loads "in the background", and then runs when the DOM is fully built. Here's the same example as above, but with `defer`: @@ -50,16 +50,18 @@ Here's the same example as above, but with `defer`: <p>...content after script...</p> ``` +In other words: + - Scripts with `defer` never block the page. -- Scripts with `defer` always execute when the DOM is ready, but before `DOMContentLoaded` event. +- Scripts with `defer` always execute when the DOM is ready (but before `DOMContentLoaded` event). -The following example demonstrates that: +The following example demonstrates the second part: ```html run height=100 <p>...content before scripts...</p> <script> - document.addEventListener('DOMContentLoaded', () => alert("DOM ready after defer!")); // (2) + document.addEventListener('DOMContentLoaded', () => alert("DOM ready after defer!")); </script> <script defer src="https://javascript.info/article/script-async-defer/long.js?speed=1"></script> @@ -68,40 +70,44 @@ The following example demonstrates that: ``` 1. The page content shows up immediately. -2. `DOMContentLoaded` waits for the deferred script. It only triggers when the script `(2)` is downloaded is executed. +2. `DOMContentLoaded` event handler waits for the deferred script. It only triggers when the script is downloaded and executed. -Deferred scripts keep their relative order, just like regular scripts. +**Deferred scripts keep their relative order, just like regular scripts.** -So, if we have a long script first, and then a smaller one, then the latter one waits. +Let's say, we have two deferred scripts: the `long.js` and then `small.js`: ```html <script defer src="https://javascript.info/article/script-async-defer/long.js"></script> <script defer src="https://javascript.info/article/script-async-defer/small.js"></script> ``` -```smart header="The small script downloads first, runs second" -Browsers scan the page for scripts and download them in parallel, to improve performance. So in the example above both scripts download in parallel. The `small.js` probably makes it first. +Browsers scan the page for scripts and download them in parallel, to improve performance. So in the example above both scripts download in parallel. The `small.js` probably finishes first. -But the specification requires scripts to execute in the document order, so it waits for `long.js` to execute. -``` +...But the `defer` attribute, besides telling the browser "not to block", ensures that the relative order is kept. So even though `small.js` loads first, it still waits and runs after `long.js` executes. + +That may be important for cases when we need to load a JavaScript library and then a script that depends on it. ```smart header="The `defer` attribute is only for external scripts" -The `defer` attribute is ignored if the script has no `src`. +The `defer` attribute is ignored if the `<script>` tag has no `src`. ``` - ## async -The `async` attribute means that a script is completely independant: +The `async` attribute is somewhat like `defer`. It also makes the script non-blocking. But it has important differences in the behavior. + +The `async` attribute means that a script is completely independent: -- The page doesn't wait for async scripts, the contents is processed and displayed. -- `DOMContentLoaded` and async scripts don't wait each other: +- The browser doesn't block on `async` scripts (like `defer`). +- Other scripts don't wait for `async` scripts, and `async` scripts don't wait for them. +- `DOMContentLoaded` and async scripts don't wait for each other: - `DOMContentLoaded` may happen both before an async script (if an async script finishes loading after the page is complete) - ...or after an async script (if an async script is short or was in HTTP-cache) -- Other scripts don't wait for `async` scripts, and `async` scripts don't wait for them. +In other words, `async` scripts load in the background and run when ready. The DOM and other scripts don't wait for them, and they don't wait for anything. A fully independent script that runs when loaded. As simple, as it can get, right? + +Here's an example similar to what we've seen with `defer`: two scripts `long.js` and `small.js`, but now with `async` instead of `defer`. -So, if we have several `async` scripts, they may execute in any order. Whatever loads first -- runs first: +They don't wait for each other. Whatever loads first (probably `small.js`) -- runs first: ```html run height=100 <p>...content before scripts...</p> @@ -116,20 +122,26 @@ So, if we have several `async` scripts, they may execute in any order. Whatever <p>...content after scripts...</p> ``` -1. The page content shows up immediately: `async` doesn't block it. -2. `DOMContentLoaded` may happen both before and after `async`, no guarantees here. -3. Async scripts don't wait for each other. A smaller script `small.js` goes second, but probably loads before `long.js`, so runs first. That's called a "load-first" order. +- The page content shows up immediately: `async` doesn't block it. +- `DOMContentLoaded` may happen both before and after `async`, no guarantees here. +- A smaller script `small.js` goes second, but probably loads before `long.js`, so `small.js` runs first. Although, it might be that `long.js` loads first, if cached, then it runs first. In other words, async scripts run in the "load-first" order. -Async scripts are great when we integrate an independant third-party script into the page: counters, ads and so on. +Async scripts are great when we integrate an independent third-party script into the page: counters, ads and so on, as they don't depend on our scripts, and our scripts shouldn't wait for them: ```html +<!-- Google Analytics is usually added like this --> <script async src="https://google-analytics.com/analytics.js"></script> ``` +```smart header="The `async` attribute is only for external scripts" +Just like `defer`, the `async` attribute is ignored if the `<script>` tag has no `src`. +``` ## Dynamic scripts -We can also create a script dynamically using JavaScript: +There's one more important way of adding a script to the page. + +We can create a script and append it to the document dynamically using JavaScript: ```js run let script = document.createElement('script'); @@ -145,21 +157,11 @@ That is: - They don't wait for anything, nothing waits for them. - The script that loads first -- runs first ("load-first" order). -We can change the load-first order into the document order by explicitly setting `async` to `false`: - -```js run -let script = document.createElement('script'); -script.src = "/article/script-async-defer/long.js"; - -*!* -script.async = false; -*/!* - -document.body.append(script); -``` +This can be changed if we explicitly set `script.async=false`. Then scripts will be executed in the document order, just like `defer`. -For example, here we add two scripts. Without `script.async=false` they would execute in load-first order (the `small.js` probably first). But with that flag the order is "as in the document": +In this example, `loadScript(src)` function adds a script and also sets `async` to `false`. +So `long.js` always runs first (as it's added first): ```js run function loadScript(src) { @@ -174,24 +176,30 @@ loadScript("/article/script-async-defer/long.js"); loadScript("/article/script-async-defer/small.js"); ``` +Without `script.async=false`, scripts would execute in default, load-first order (the `small.js` probably first). + +Again, as with the `defer`, the order matters if we'd like to load a library and then another script that depends on it. + ## Summary -Both `async` and `defer` have one common thing: they don't block page rendering. So the user can read page content and get acquanted with the page immediately. +Both `async` and `defer` have one common thing: downloading of such scripts doesn't block page rendering. So the user can read page content and get acquainted with the page immediately. But there are also essential differences between them: | | Order | `DOMContentLoaded` | |---------|---------|---------| -| `async` | *Load-first order*. Their document order doesn't matter -- which loads first | Irrelevant. May load and execute while the document has not yet been fully downloaded. That happens if scripts are small or cached, and the document is long enough. | +| `async` | *Load-first order*. Their document order doesn't matter -- which loads first runs first | Irrelevant. May load and execute while the document has not yet been fully downloaded. That happens if scripts are small or cached, and the document is long enough. | | `defer` | *Document order* (as they go in the document). | Execute after the document is loaded and parsed (they wait if needed), right before `DOMContentLoaded`. | -```warn header="Page without scripts should be usable" -Please note that if you're using `defer`, then the page is visible before the script loads and enables all the graphical components. +In practice, `defer` is used for scripts that need the whole DOM and/or their relative execution order is important. -So, buttons should be disabled by CSS or by other means, to let the user +And `async` is used for independent scripts, like counters or ads. And their relative execution order does not matter. -In practice, `defer` is used for scripts that need DOM and/or their relative execution order is important. +```warn header="Page without scripts should be usable" +Please note: if you're using `defer` or `async`, then user will see the page *before* the script loads. +In such case, some graphical components are probably not initialized yet. -So `async` is used for independent scripts, like counters or ads, that don't need to access page content. And their relative execution order does not matter. +Don't forget to put "loading" indication and disable buttons that aren't functional yet. Let the user clearly see what he can do on the page, and what's still getting ready. +``` diff --git a/2-ui/5-loading/02-script-async-defer/window-onbeforeunload.view/index.html b/2-ui/5-loading/02-script-async-defer/window-onbeforeunload.view/index.html deleted file mode 100644 index eacfda1f7..000000000 --- a/2-ui/5-loading/02-script-async-defer/window-onbeforeunload.view/index.html +++ /dev/null @@ -1,19 +0,0 @@ -<!DOCTYPE html> -<html> -<head> - <meta charset="utf-8"> -</head> -<body> - <script> - function setHandler() { - window.onbeforeunload = function() { - return "There are unsaved changes. Leave now?"; - }; - } - </script> - - <button onclick="setHandler()">Set window.onbeforeunload</button> - - <a href="http://example.com">Leave for EXAMPLE.COM</a> -</body> -</html> diff --git a/2-ui/5-loading/03-onload-onerror/1-load-img-callback/solution.view/index.html b/2-ui/5-loading/03-onload-onerror/1-load-img-callback/solution.view/index.html index c1624d038..50b9e741d 100644 --- a/2-ui/5-loading/03-onload-onerror/1-load-img-callback/solution.view/index.html +++ b/2-ui/5-loading/03-onload-onerror/1-load-img-callback/solution.view/index.html @@ -35,7 +35,7 @@ } // for each image, - // let's create another img with the same src and check that we have its width immediately + // let's create another img with the same src and check that we have its width function testLoaded() { let widthSum = 0; for (let i = 0; i < sources.length; i++) { diff --git a/2-ui/5-loading/03-onload-onerror/article.md b/2-ui/5-loading/03-onload-onerror/article.md index e915dedb6..590e54ab4 100644 --- a/2-ui/5-loading/03-onload-onerror/article.md +++ b/2-ui/5-loading/03-onload-onerror/article.md @@ -1,6 +1,6 @@ # Resource loading: onload and onerror -The browser allows to track the loading of external resources -- scripts, iframes, pictures and so on. +The browser allows us to track the loading of external resources -- scripts, iframes, pictures and so on. There are two events for it: @@ -41,19 +41,19 @@ document.head.append(script); *!* script.onload = function() { - // the script creates a helper function "_" - alert(_); // the function is available + // the script creates a variable "_" + alert( _.VERSION ); // shows library version }; */!* ``` So in `onload` we can use script variables, run functions etc. -...And what if the loading failed? For instance, there's no such script (error 404) or the server or the server is down (unavailable). +...And what if the loading failed? For instance, there's no such script (error 404) or the server is down (unavailable). ### script.onerror -Errors that occur during the loading of the script can be tracked on `error` event. +Errors that occur during the loading of the script can be tracked in an `error` event. For instance, let's request a script that doesn't exist: @@ -69,12 +69,12 @@ script.onerror = function() { */!* ``` -Please note that we can't get HTTP error details here. We don't know was it error 404 or 500 or something else. Just that the loading failed. +Please note that we can't get HTTP error details here. We don't know if it was an error 404 or 500 or something else. Just that the loading failed. ```warn Events `onload`/`onerror` track only the loading itself. -Errors during script processing and execution are out of the scope of these events. To track script errors, one can use `window.onerror` global handler. +Errors that may occur during script processing and execution are out of scope for these events. That is: if a script loaded successfully, then `onload` triggers, even if it has programming errors in it. To track script errors, one can use `window.onerror` global handler. ``` ## Other resources @@ -92,13 +92,13 @@ img.onload = function() { }; img.onerror = function() { - alert("Error occured while loading image"); + alert("Error occurred while loading image"); }; ``` There are some notes though: -- Most resources start loading when they are added to the document. But `<img>` is an exception. It starts loading when it gets an src `(*)`. +- Most resources start loading when they are added to the document. But `<img>` is an exception. It starts loading when it gets a src `(*)`. - For `<iframe>`, the `iframe.onload` event triggers when the iframe loading finished, both for successful load and in case of an error. That's for historical reasons. @@ -107,19 +107,19 @@ That's for historical reasons. There's a rule: scripts from one site can't access contents of the other site. So, e.g. a script at `https://facebook.com` can't read the user's mailbox at `https://gmail.com`. -Or, to be more precise, one origin (domain/port/protocol triplet) can't access the content from another one. So even if we have a subdomain, or just another port, these are different origins, no access to each other. +Or, to be more precise, one origin (domain/port/protocol triplet) can't access the content from another one. So even if we have a subdomain, or just another port, these are different origins with no access to each other. This rule also affects resources from other domains. If we're using a script from another domain, and there's an error in it, we can't get error details. -For example, let's take a script with a single (bad) function call: +For example, let's take a script `error.js` that consists of a single (bad) function call: ```js // 📁 error.js noSuchFunction(); ``` -Now load it from our domain: +Now load it from the same site where it's located: ```html run height=0 <script> @@ -155,24 +155,24 @@ Script error. , 0:0 ``` -Details may vary depeding on the browser, but the idea is same: any information about the internals of a script is hidden. Exactly because it's from another domain. +Details may vary depending on the browser, but the idea is the same: any information about the internals of a script, including error stack traces, is hidden. Exactly because it's from another domain. -Why do we need the details? +Why do we need error details? -There are many services (and we can build our own) that listen to `window.onerror`, save errors at the server and provide an interface to access and analyze them. That's great, as we can see real errors, triggered by our users. But we can't see any error information for scripts from other domains. +There are many services (and we can build our own) that listen for global errors using `window.onerror`, save errors and provide an interface to access and analyze them. That's great, as we can see real errors, triggered by our users. But if a script comes from another origin, then there's not much information about errors in it, as we've just seen. Similar cross-origin policy (CORS) is enforced for other types of resources as well. -**To allow cross-origin access, we need `crossorigin` attribute, plus the remote server must provide special headers.** +**To allow cross-origin access, the `<script>` tag needs to have the `crossorigin` attribute, plus the remote server must provide special headers.** There are three levels of cross-origin access: 1. **No `crossorigin` attribute** -- access prohibited. 2. **`crossorigin="anonymous"`** -- access allowed if the server responds with the header `Access-Control-Allow-Origin` with `*` or our origin. Browser does not send authorization information and cookies to remote server. -3. **`crossorigin="use-credentials"`** -- access allowed if the server sends back the header `Access-Control-Allow-Origin` with our origin and `Access-Control-Allow-Credentials: true`. Browser sends authorization information and cookies to remote server. +3. **`crossorigin="use-credentials"`** -- access allowed if the server sends back the header `Access-Control-Allow-Origin` with our origin and `Access-Control-Allow-Credentials: true`. Browser sends authorization information and cookies to remote server. ```smart -You can read more about cross-origin access in the chapter <info:fetch-crossorigin>. It describes `fetch` method for network requests, but the policy is exactly the same. +You can read more about cross-origin access in the chapter <info:fetch-crossorigin>. It describes the `fetch` method for network requests, but the policy is exactly the same. Such thing as "cookies" is out of our current scope, but you can read about them in the chapter <info:cookie>. ``` @@ -181,7 +181,7 @@ In our case, we didn't have any crossorigin attribute. So the cross-origin acces We can choose between `"anonymous"` (no cookies sent, one server-side header needed) and `"use-credentials"` (sends cookies too, two server-side headers needed). -If we don't care about cookies, then `"anonymous"` is a way to go: +If we don't care about cookies, then `"anonymous"` is the way to go: ```html run height=0 <script> @@ -192,7 +192,7 @@ window.onerror = function(message, url, line, col, errorObj) { <script *!*crossorigin="anonymous"*/!* src="https://cors.javascript.info/article/onload-onerror/crossorigin/error.js"></script> ``` -Now, assuming that the server provides `Access-Control-Allow-Origin` header, everything's fine. We have the full error report. +Now, assuming that the server provides an `Access-Control-Allow-Origin` header, everything's fine. We have the full error report. ## Summary diff --git a/2-ui/99-ui-misc/01-mutation-observer/article.md b/2-ui/99-ui-misc/01-mutation-observer/article.md new file mode 100644 index 000000000..3cf6f5397 --- /dev/null +++ b/2-ui/99-ui-misc/01-mutation-observer/article.md @@ -0,0 +1,273 @@ + +# Mutation observer + +`MutationObserver` is a built-in object that observes a DOM element and fires a callback when it detects a change. + +We'll first take a look at the syntax, and then explore a real-world use case, to see where such thing may be useful. + +## Syntax + +`MutationObserver` is easy to use. + +First, we create an observer with a callback-function: + +```js +let observer = new MutationObserver(callback); +``` + +And then attach it to a DOM node: + +```js +observer.observe(node, config); +``` + +`config` is an object with boolean options "what kind of changes to react on": +- `childList` -- changes in the direct children of `node`, +- `subtree` -- in all descendants of `node`, +- `attributes` -- attributes of `node`, +- `attributeFilter` -- an array of attribute names, to observe only selected ones. +- `characterData` -- whether to observe `node.data` (text content), + +Few other options: +- `attributeOldValue` -- if `true`, pass both the old and the new value of attribute to callback (see below), otherwise only the new one (needs `attributes` option), +- `characterDataOldValue` -- if `true`, pass both the old and the new value of `node.data` to callback (see below), otherwise only the new one (needs `characterData` option). + +Then after any changes, the `callback` is executed: changes are passed in the first argument as a list of [MutationRecord](https://dom.spec.whatwg.org/#mutationrecord) objects, and the observer itself as the second argument. + +[MutationRecord](https://dom.spec.whatwg.org/#mutationrecord) objects have properties: + +- `type` -- mutation type, one of + - `"attributes"`: attribute modified + - `"characterData"`: data modified, used for text nodes, + - `"childList"`: child elements added/removed, +- `target` -- where the change occurred: an element for `"attributes"`, or text node for `"characterData"`, or an element for a `"childList"` mutation, +- `addedNodes/removedNodes` -- nodes that were added/removed, +- `previousSibling/nextSibling` -- the previous and next sibling to added/removed nodes, +- `attributeName/attributeNamespace` -- the name/namespace (for XML) of the changed attribute, +- `oldValue` -- the previous value, only for attribute or text changes, if the corresponding option is set `attributeOldValue`/`characterDataOldValue`. + +For example, here's a `<div>` with a `contentEditable` attribute. That attribute allows us to focus on it and edit. + +```html run +<div contentEditable id="elem">Click and <b>edit</b>, please</div> + +<script> +let observer = new MutationObserver(mutationRecords => { + console.log(mutationRecords); // console.log(the changes) +}); + +// observe everything except attributes +observer.observe(elem, { + childList: true, // observe direct children + subtree: true, // and lower descendants too + characterDataOldValue: true // pass old data to callback +}); +</script> +``` + +If we run this code in the browser, then focus on the given `<div>` and change the text inside `<b>edit</b>`, `console.log` will show one mutation: + +```js +mutationRecords = [{ + type: "characterData", + oldValue: "edit", + target: <text node>, + // other properties empty +}]; +``` + +If we make more complex editing operations, e.g. remove the `<b>edit</b>`, the mutation event may contain multiple mutation records: + +```js +mutationRecords = [{ + type: "childList", + target: <div#elem>, + removedNodes: [<b>], + nextSibling: <text node>, + previousSibling: <text node> + // other properties empty +}, { + type: "characterData" + target: <text node> + // ...mutation details depend on how the browser handles such removal + // it may coalesce two adjacent text nodes "edit " and ", please" into one node + // or it may leave them separate text nodes +}]; +``` + +So, `MutationObserver` allows to react on any changes within DOM subtree. + +## Usage for integration + +When such thing may be useful? + +Imagine the situation when you need to add a third-party script that contains useful functionality, but also does something unwanted, e.g. shows ads `<div class="ads">Unwanted ads</div>`. + +Naturally, the third-party script provides no mechanisms to remove it. + +Using `MutationObserver`, we can detect when the unwanted element appears in our DOM and remove it. + +There are other situations when a third-party script adds something into our document, and we'd like to detect, when it happens, to adapt our page, dynamically resize something etc. + +`MutationObserver` allows to implement this. + +## Usage for architecture + +There are also situations when `MutationObserver` is good from architectural standpoint. + +Let's say we're making a website about programming. Naturally, articles and other materials may contain source code snippets. + +Such snippet in an HTML markup looks like this: + +```html +... +<pre class="language-javascript"><code> + // here's the code + let hello = "world"; +</code></pre> +... +``` + +For better readability and at the same time, to beautify it, we'll be using a JavaScript syntax highlighting library on our site, like [Prism.js](https://prismjs.com/). To get syntax highlighting for above snippet in Prism, `Prism.highlightElem(pre)` is called, which examines the contents of such `pre` elements and adds special tags and styles for colored syntax highlighting into those elements, similar to what you see in examples here, on this page. + +When exactly should we run that highlighting method? Well, we can do it on `DOMContentLoaded` event, or put the script at the bottom of the page. The moment our DOM is ready, we can search for elements `pre[class*="language"]` and call `Prism.highlightElem` on them: + +```js +// highlight all code snippets on the page +document.querySelectorAll('pre[class*="language"]').forEach(Prism.highlightElem); +``` + +Everything's simple so far, right? We find code snippets in HTML and highlight them. + +Now let's go on. Let's say we're going to dynamically fetch materials from a server. We'll study methods for that [later in the tutorial](info:fetch). For now it only matters that we fetch an HTML article from a webserver and display it on demand: + +```js +let article = /* fetch new content from server */ +articleElem.innerHTML = article; +``` + +The new `article` HTML may contain code snippets. We need to call `Prism.highlightElem` on them, otherwise they won't get highlighted. + +**Where and when to call `Prism.highlightElem` for a dynamically loaded article?** + +We could append that call to the code that loads an article, like this: + +```js +let article = /* fetch new content from server */ +articleElem.innerHTML = article; + +*!* +let snippets = articleElem.querySelectorAll('pre[class*="language-"]'); +snippets.forEach(Prism.highlightElem); +*/!* +``` + +...But, imagine if we have many places in the code where we load our content - articles, quizzes, forum posts, etc. Do we need to put the highlighting call everywhere, to highlight the code in content after loading? That's not very convenient. + +And what if the content is loaded by a third-party module? For example, we have a forum written by someone else, that loads content dynamically, and we'd like to add syntax highlighting to it. No one likes patching third-party scripts. + +Luckily, there's another option. + +We can use `MutationObserver` to automatically detect when code snippets are inserted into the page and highlight them. + +So we'll handle the highlighting functionality in one place, relieving us from the need to integrate it. + +### Dynamic highlight demo + +Here's the working example. + +If you run this code, it starts observing the element below and highlighting any code snippets that appear there: + +```js run +let observer = new MutationObserver(mutations => { + + for(let mutation of mutations) { + // examine new nodes, is there anything to highlight? + + for(let node of mutation.addedNodes) { + // we track only elements, skip other nodes (e.g. text nodes) + if (!(node instanceof HTMLElement)) continue; + + // check the inserted element for being a code snippet + if (node.matches('pre[class*="language-"]')) { + Prism.highlightElement(node); + } + + // or maybe there's a code snippet somewhere in its subtree? + for(let elem of node.querySelectorAll('pre[class*="language-"]')) { + Prism.highlightElement(elem); + } + } + } + +}); + +let demoElem = document.getElementById('highlight-demo'); + +observer.observe(demoElem, {childList: true, subtree: true}); +``` + +Here, below, there's an HTML-element and JavaScript that dynamically fills it using `innerHTML`. + +Please run the previous code (above, observes that element), and then the code below. You'll see how `MutationObserver` detects and highlights the snippet. + +<p id="highlight-demo" style="border: 1px solid #ddd">A demo-element with <code>id="highlight-demo"</code>, run the code above to observe it.</p> + +The following code populates its `innerHTML`, that causes the `MutationObserver` to react and highlight its contents: + +```js run +let demoElem = document.getElementById('highlight-demo'); + +// dynamically insert content with code snippets +demoElem.innerHTML = `A code snippet is below: + <pre class="language-javascript"><code> let hello = "world!"; </code></pre> + <div>Another one:</div> + <div> + <pre class="language-css"><code>.class { margin: 5px; } </code></pre> + </div> +`; +``` + +Now we have `MutationObserver` that can track all highlighting in observed elements or the whole `document`. We can add/remove code snippets in HTML without thinking about it. + +## Additional methods + +There's a method to stop observing the node: + +- `observer.disconnect()` -- stops the observation. + +When we stop the observing, it might be possible that some changes were not yet processed by the observer. In such cases, we use + +- `observer.takeRecords()` -- gets a list of unprocessed mutation records - those that happened, but the callback has not handled them. + +These methods can be used together, like this: + +```js +// get a list of unprocessed mutations +// should be called before disconnecting, +// if you care about possibly unhandled recent mutations +let mutationRecords = observer.takeRecords(); + +// stop tracking changes +observer.disconnect(); +... +``` + + +```smart header="Records returned by `observer.takeRecords()` are removed from the processing queue" +The callback won't be called for records, returned by `observer.takeRecords()`. +``` + +```smart header="Garbage collection interaction" +Observers use weak references to nodes internally. That is, if a node is removed from the DOM, and becomes unreachable, then it can be garbage collected. + +The mere fact that a DOM node is observed doesn't prevent the garbage collection. +``` + +## Summary + +`MutationObserver` can react to changes in DOM - attributes, text content and adding/removing elements. + +We can use it to track changes introduced by other parts of our code, as well as to integrate with third-party scripts. + +`MutationObserver` can track any changes. The config "what to observe" options are used for optimizations, not to spend resources on unneeded callback invocations. diff --git a/2-ui/99-ui-misc/02-selection-range/article.md b/2-ui/99-ui-misc/02-selection-range/article.md new file mode 100644 index 000000000..09a20bc67 --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/article.md @@ -0,0 +1,716 @@ +libs: + - d3 + - domtree + +--- + +# Selection and Range + +In this chapter we'll cover selection in the document, as well as selection in form fields, such as `<input>`. + +JavaScript can access an existing selection, select/deselect DOM nodes as a whole or partially, remove the selected content from the document, wrap it into a tag, and so on. + +You can find some recipes for common tasks at the end of the chapter, in "Summary" section. Maybe that covers your current needs, but you'll get much more if you read the whole text. + +The underlying `Range` and `Selection` objects are easy to grasp, and then you'll need no recipes to make them do what you want. + +## Range + +The basic concept of selection is [Range](https://dom.spec.whatwg.org/#ranges), that is essentially a pair of "boundary points": range start and range end. + +A `Range` object is created without parameters: + +```js +let range = new Range(); +``` + +Then we can set the selection boundaries using `range.setStart(node, offset)` and `range.setEnd(node, offset)`. + +As you might guess, further we'll use the `Range` objects for selection, but first let's create few such objects. + +### Selecting the text partially + +The interesting thing is that the first argument `node` in both methods can be either a text node or an element node, and the meaning of the second argument depends on that. + +**If `node` is a text node, then `offset` must be the position in its text.** + +For example, given the element `<p>Hello</p>`, we can create the range containing the letters "ll" as follows: + +```html run +<p id="p">Hello</p> +<script> + let range = new Range(); + range.setStart(p.firstChild, 2); + range.setEnd(p.firstChild, 4); + + // toString of a range returns its content as text + console.log(range); // ll +</script> +``` + +Here we take the first child of `<p>` (that's the text node) and specify the text positions inside it: + + + +### Selecting element nodes + +**Alternatively, if `node` is an element node, then `offset` must be the child number.** + +That's handy for making ranges that contain nodes as a whole, not stop somewhere inside their text. + +For example, we have a more complex document fragment: + +```html autorun +<p id="p">Example: <i>italic</i> and <b>bold</b></p> +``` + +Here's its DOM structure with both element and text nodes: + +<div class="select-p-domtree"></div> + +<script> +let selectPDomtree = { + "name": "P", + "nodeType": 1, + "children": [{ + "name": "#text", + "nodeType": 3, + "content": "Example: " + }, { + "name": "I", + "nodeType": 1, + "children": [{ + "name": "#text", + "nodeType": 3, + "content": "italic" + }] + }, { + "name": "#text", + "nodeType": 3, + "content": " and " + }, { + "name": "B", + "nodeType": 1, + "children": [{ + "name": "#text", + "nodeType": 3, + "content": "bold" + }] + }] +} + +drawHtmlTree(selectPDomtree, 'div.select-p-domtree', 690, 320); +</script> + +Let's make a range for `"Example: <i>italic</i>"`. + +As we can see, this phrase consists of exactly two children of `<p>`, with indexes `0` and `1`: + + + +- The starting point has `<p>` as the parent `node`, and `0` as the offset. + + So we can set it as `range.setStart(p, 0)`. +- The ending point also has `<p>` as the parent `node`, but `2` as the offset (it specifies the range up to, but not including `offset`). + + So we can set it as `range.setEnd(p, 2)`. + +Here's the demo. If you run it, you can see that the text gets selected: + +```html run +<p id="p">Example: <i>italic</i> and <b>bold</b></p> + +<script> +*!* + let range = new Range(); + + range.setStart(p, 0); + range.setEnd(p, 2); +*/!* + + // toString of a range returns its content as text, without tags + console.log(range); // Example: italic + + // apply this range for document selection (explained later below) + document.getSelection().addRange(range); +</script> +``` + +Here's a more flexible test stand where you can set range start/end numbers and explore other variants: + +```html run autorun +<p id="p">Example: <i>italic</i> and <b>bold</b></p> + +From <input id="start" type="number" value=1> – To <input id="end" type="number" value=4> +<button id="button">Click to select</button> +<script> + button.onclick = () => { + *!* + let range = new Range(); + + range.setStart(p, start.value); + range.setEnd(p, end.value); + */!* + + // apply the selection, explained later below + document.getSelection().removeAllRanges(); + document.getSelection().addRange(range); + }; +</script> +``` + +E.g. selecting in the same `<p>` from offset `1` to `4` gives us the range `<i>italic</i> and <b>bold</b>`: + + + +```smart header="Starting and ending nodes can be different" +We don't have to use the same node in `setStart` and `setEnd`. A range may span across many unrelated nodes. It's only important that the end is after the start in the document. +``` + +### Selecting a bigger fragment + +Let's make a bigger selection in our example, like this: + + + +We already know how to do that. We just need to set the start and the end as a relative offset in text nodes. + +We need to create a range, that: +- starts from position 2 in `<p>` first child (taking all but two first letters of "Ex<b>ample:</b> ") +- ends at the position 3 in `<b>` first child (taking first three letters of "<b>bol</b>d", but no more): + +```html run +<p id="p">Example: <i>italic</i> and <b>bold</b></p> + +<script> + let range = new Range(); + + range.setStart(p.firstChild, 2); + range.setEnd(p.querySelector('b').firstChild, 3); + + console.log(range); // ample: italic and bol + + // use this range for selection (explained later) + window.getSelection().addRange(range); +</script> +``` + +As you can see, it's fairly easy to make a range of whatever we want. + +If we'd like to take nodes as a whole, we can pass elements in `setStart/setEnd`. Otherwise, we can work on the text level. + +## Range properties + +The range object that we created in the example above has following properties: + + + +- `startContainer`, `startOffset` -- node and offset of the start, + - in the example above: first text node inside `<p>` and `2`. +- `endContainer`, `endOffset` -- node and offset of the end, + - in the example above: first text node inside `<b>` and `3`. +- `collapsed` -- boolean, `true` if the range starts and ends on the same point (so there's no content inside the range), + - in the example above: `false` +- `commonAncestorContainer` -- the nearest common ancestor of all nodes within the range, + - in the example above: `<p>` + + +## Range selection methods + +There are many convenient methods to manipulate ranges. + +We've already seen `setStart` and `setEnd`, here are other similar methods. + +Set range start: + +- `setStart(node, offset)` set start at: position `offset` in `node` +- `setStartBefore(node)` set start at: right before `node` +- `setStartAfter(node)` set start at: right after `node` + +Set range end (similar methods): + +- `setEnd(node, offset)` set end at: position `offset` in `node` +- `setEndBefore(node)` set end at: right before `node` +- `setEndAfter(node)` set end at: right after `node` + +Technically, `setStart/setEnd` can do anything, but more methods provide more convenience. + +In all these methods, `node` can be both a text or element node: for text nodes `offset` skips that many of characters, while for element nodes that many child nodes. + +Even more methods to create ranges: +- `selectNode(node)` set range to select the whole `node` +- `selectNodeContents(node)` set range to select the whole `node` contents +- `collapse(toStart)` if `toStart=true` set end=start, otherwise set start=end, thus collapsing the range +- `cloneRange()` creates a new range with the same start/end + +## Range editing methods + +Once the range is created, we can manipulate its content using these methods: + +- `deleteContents()` -- remove range content from the document +- `extractContents()` -- remove range content from the document and return as [DocumentFragment](info:modifying-document#document-fragment) +- `cloneContents()` -- clone range content and return as [DocumentFragment](info:modifying-document#document-fragment) +- `insertNode(node)` -- insert `node` into the document at the beginning of the range +- `surroundContents(node)` -- wrap `node` around range content. For this to work, the range must contain both opening and closing tags for all elements inside it: no partial ranges like `<i>abc`. + +With these methods we can do basically anything with selected nodes. + +Here's the test stand to see them in action: + +```html run refresh autorun height=260 +Click buttons to run methods on the selection, "resetExample" to reset it. + +<p id="p">Example: <i>italic</i> and <b>bold</b></p> + +<p id="result"></p> +<script> + let range = new Range(); + + // Each demonstrated method is represented here: + let methods = { + deleteContents() { + range.deleteContents() + }, + extractContents() { + let content = range.extractContents(); + result.innerHTML = ""; + result.append("extracted: ", content); + }, + cloneContents() { + let content = range.cloneContents(); + result.innerHTML = ""; + result.append("cloned: ", content); + }, + insertNode() { + let newNode = document.createElement('u'); + newNode.innerHTML = "NEW NODE"; + range.insertNode(newNode); + }, + surroundContents() { + let newNode = document.createElement('u'); + try { + range.surroundContents(newNode); + } catch(e) { console.log(e) } + }, + resetExample() { + p.innerHTML = `Example: <i>italic</i> and <b>bold</b>`; + result.innerHTML = ""; + + range.setStart(p.firstChild, 2); + range.setEnd(p.querySelector('b').firstChild, 3); + + window.getSelection().removeAllRanges(); + window.getSelection().addRange(range); + } + }; + + for(let method in methods) { + document.write(`<div><button onclick="methods.${method}()">${method}</button></div>`); + } + + methods.resetExample(); +</script> +``` + +There also exist methods to compare ranges, but these are rarely used. When you need them, please refer to the [spec](https://dom.spec.whatwg.org/#interface-range) or [MDN manual](mdn:/api/Range). + + +## Selection + +`Range` is a generic object for managing selection ranges. Although, creating a `Range` doesn't mean that we see a selection on screen. + +We may create `Range` objects, pass them around -- they do not visually select anything on their own. + +The document selection is represented by `Selection` object, that can be obtained as `window.getSelection()` or `document.getSelection()`. A selection may include zero or more ranges. At least, the [Selection API specification](https://www.w3.org/TR/selection-api/) says so. In practice though, only Firefox allows to select multiple ranges in the document by using `key:Ctrl+click` (`key:Cmd+click` for Mac). + +Here's a screenshot of a selection with 3 ranges, made in Firefox: + + + +Other browsers support at maximum 1 range. As we'll see, some of `Selection` methods imply that there may be many ranges, but again, in all browsers except Firefox, there's at maximum 1. + +Here's a small demo that shows the current selection (select something and click) as text: + +<button onclick="alert(document.getSelection())">alert(document.getSelection())</button> + +## Selection properties + +As said, a selection may in theory contain multiple ranges. We can get these range objects using the method: + +- `getRangeAt(i)` -- get i-th range, starting from `0`. In all browsers except Firefox, only `0` is used. + +Also, there exist properties that often provide better convenience. + +Similar to a range, a selection object has a start, called "anchor", and the end, called "focus". + +The main selection properties are: + +- `anchorNode` -- the node where the selection starts, +- `anchorOffset` -- the offset in `anchorNode` where the selection starts, +- `focusNode` -- the node where the selection ends, +- `focusOffset` -- the offset in `focusNode` where the selection ends, +- `isCollapsed` -- `true` if selection selects nothing (empty range), or doesn't exist. +- `rangeCount` -- count of ranges in the selection, maximum `1` in all browsers except Firefox. + +```smart header="Selection end/start vs Range" + +There's an important difference between a selection anchor/focus compared with a `Range` start/end. + +As we know, `Range` objects always have their start before the end. + +For selections, that's not always the case. + +Selecting something with a mouse can be done in both directions: either "left-to-right" or "right-to-left". + +In other words, when the mouse button is pressed, and then it moves forward in the document, then its end (focus) will be after its start (anchor). + +E.g. if the user starts selecting with mouse and goes from "Example" to "italic": + + + +...But the same selection could be done backwards: starting from "italic" to "Example" (backward direction), then its end (focus) will be before the start (anchor): + + +``` + +## Selection events + +There are events on to keep track of selection: + +- `elem.onselectstart` -- when a selection *starts* specifically on element `elem` (or inside it). For instance, when the user presses the mouse button on it and starts to move the pointer. + - Preventing the default action cancels the selection start. So starting a selection from this element becomes impossible, but the element is still selectable. The visitor just needs to start the selection from elsewhere. +- `document.onselectionchange` -- whenever a selection changes or starts. + - Please note: this handler can be set only on `document`, it tracks all selections in it. + +### Selection tracking demo + +Here's a small demo. It tracks the current selection on the `document` and shows its boundaries: + +```html run height=80 +<p id="p">Select me: <i>italic</i> and <b>bold</b></p> + +From <input id="from" disabled> – To <input id="to" disabled> +<script> + document.onselectionchange = function() { + let selection = document.getSelection(); + + let {anchorNode, anchorOffset, focusNode, focusOffset} = selection; + + // anchorNode and focusNode are text nodes usually + from.value = `${anchorNode?.data}, offset ${anchorOffset}`; + to.value = `${focusNode?.data}, offset ${focusOffset}`; + }; +</script> +``` + +### Selection copying demo + +There are two approaches to copying the selected content: + +1. We can use `document.getSelection().toString()` to get it as text. +2. Otherwise, to copy the full DOM, e.g. if we need to keep formatting, we can get the underlying ranges with `getRangeAt(...)`. A `Range` object, in turn, has `cloneContents()` method that clones its content and returns as `DocumentFragment` object, that we can insert elsewhere. + +Here's the demo of copying the selected content both as text and as DOM nodes: + +```html run height=100 +<p id="p">Select me: <i>italic</i> and <b>bold</b></p> + +Cloned: <span id="cloned"></span> +<br> +As text: <span id="astext"></span> + +<script> + document.onselectionchange = function() { + let selection = document.getSelection(); + + cloned.innerHTML = astext.innerHTML = ""; + + // Clone DOM nodes from ranges (we support multiselect here) + for (let i = 0; i < selection.rangeCount; i++) { + cloned.append(selection.getRangeAt(i).cloneContents()); + } + + // Get as text + astext.innerHTML += selection; + }; +</script> +``` + +## Selection methods + +We can work with the selection by adding/removing ranges: + +- `getRangeAt(i)` -- get i-th range, starting from `0`. In all browsers except Firefox, only `0` is used. +- `addRange(range)` -- add `range` to selection. All browsers except Firefox ignore the call, if the selection already has an associated range. +- `removeRange(range)` -- remove `range` from the selection. +- `removeAllRanges()` -- remove all ranges. +- `empty()` -- alias to `removeAllRanges`. + +There are also convenience methods to manipulate the selection range directly, without intermediate `Range` calls: + +- `collapse(node, offset)` -- replace selected range with a new one that starts and ends at the given `node`, at position `offset`. +- `setPosition(node, offset)` -- alias to `collapse`. +- `collapseToStart()` - collapse (replace with an empty range) to selection start, +- `collapseToEnd()` - collapse to selection end, +- `extend(node, offset)` - move focus of the selection to the given `node`, position `offset`, +- `setBaseAndExtent(anchorNode, anchorOffset, focusNode, focusOffset)` - replace selection range with the given start `anchorNode/anchorOffset` and end `focusNode/focusOffset`. All content in-between them is selected. +- `selectAllChildren(node)` -- select all children of the `node`. +- `deleteFromDocument()` -- remove selected content from the document. +- `containsNode(node, allowPartialContainment = false)` -- checks whether the selection contains `node` (partially if the second argument is `true`) + +For most tasks these methods are just fine, there's no need to access the underlying `Range` object. + +For example, selecting the whole contents of the paragraph `<p>`: + +```html run +<p id="p">Select me: <i>italic</i> and <b>bold</b></p> + +<script> + // select from 0th child of <p> to the last child + document.getSelection().setBaseAndExtent(p, 0, p, p.childNodes.length); +</script> +``` + +The same thing using ranges: + +```html run +<p id="p">Select me: <i>italic</i> and <b>bold</b></p> + +<script> + let range = new Range(); + range.selectNodeContents(p); // or selectNode(p) to select the <p> tag too + + document.getSelection().removeAllRanges(); // clear existing selection if any + document.getSelection().addRange(range); +</script> +``` + +```smart header="To select something, remove the existing selection first" +If a document selection already exists, empty it first with `removeAllRanges()`. And then add ranges. Otherwise, all browsers except Firefox ignore new ranges. + +The exception is some selection methods, that replace the existing selection, such as `setBaseAndExtent`. +``` + +## Selection in form controls + +Form elements, such as `input` and `textarea` provide [special API for selection](https://html.spec.whatwg.org/#textFieldSelection), without `Selection` or `Range` objects. As an input value is a pure text, not HTML, there's no need for such objects, everything's much simpler. + +Properties: +- `input.selectionStart` -- position of selection start (writeable), +- `input.selectionEnd` -- position of selection end (writeable), +- `input.selectionDirection` -- selection direction, one of: "forward", "backward" or "none" (if e.g. selected with a double mouse click), + +Events: +- `input.onselect` -- triggers when something is selected. + +Methods: + +- `input.select()` -- selects everything in the text control (can be `textarea` instead of `input`), +- `input.setSelectionRange(start, end, [direction])` -- change the selection to span from position `start` till `end`, in the given direction (optional). +- `input.setRangeText(replacement, [start], [end], [selectionMode])` -- replace a range of text with the new text. + + Optional arguments `start` and `end`, if provided, set the range start and end, otherwise user selection is used. + + The last argument, `selectionMode`, determines how the selection will be set after the text has been replaced. The possible values are: + + - `"select"` -- the newly inserted text will be selected. + - `"start"` -- the selection range collapses just before the inserted text (the cursor will be immediately before it). + - `"end"` -- the selection range collapses just after the inserted text (the cursor will be right after it). + - `"preserve"` -- attempts to preserve the selection. This is the default. + +Now let's see these methods in action. + +### Example: tracking selection + +For example, this code uses `onselect` event to track selection: + +```html run autorun +<textarea id="area" style="width:80%;height:60px"> +Selecting in this text updates values below. +</textarea> +<br> +From <input id="from" disabled> – To <input id="to" disabled> + +<script> + area.onselect = function() { + from.value = area.selectionStart; + to.value = area.selectionEnd; + }; +</script> +``` + +Please note: +- `onselect` triggers when something is selected, but not when the selection is removed. +- `document.onselectionchange` event should not trigger for selections inside a form control, according to the [spec](https://w3c.github.io/selection-api/#dfn-selectionchange), as it's not related to `document` selection and ranges. Some browsers generate it, but we shouldn't rely on it. + + +### Example: moving cursor + +We can change `selectionStart` and `selectionEnd`, that sets the selection. + +An important edge case is when `selectionStart` and `selectionEnd` equal each other. Then it's exactly the cursor position. Or, to rephrase, when nothing is selected, the selection is collapsed at the cursor position. + +So, by setting `selectionStart` and `selectionEnd` to the same value, we move the cursor. + +For example: + +```html run autorun +<textarea id="area" style="width:80%;height:60px"> +Focus on me, the cursor will be at position 10. +</textarea> + +<script> + area.onfocus = () => { + // zero delay setTimeout to run after browser "focus" action finishes + setTimeout(() => { + // we can set any selection + // if start=end, the cursor is exactly at that place + area.selectionStart = area.selectionEnd = 10; + }); + }; +</script> +``` + +### Example: modifying selection + +To modify the content of the selection, we can use `input.setRangeText()` method. Of course, we can read `selectionStart/End` and, with the knowledge of the selection, change the corresponding substring of `value`, but `setRangeText` is more powerful and often more convenient. + +That's a somewhat complex method. In its simplest one-argument form it replaces the user selected range and removes the selection. + +For example, here the user selection will be wrapped by `*...*`: + +```html run autorun +<input id="input" style="width:200px" value="Select here and click the button"> +<button id="button">Wrap selection in stars *...*</button> + +<script> +button.onclick = () => { + if (input.selectionStart == input.selectionEnd) { + return; // nothing is selected + } + + let selected = input.value.slice(input.selectionStart, input.selectionEnd); + input.setRangeText(`*${selected}*`); +}; +</script> +``` + +With more arguments, we can set range `start` and `end`. + +In this example we find `"THIS"` in the input text, replace it and keep the replacement selected: + +```html run autorun +<input id="input" style="width:200px" value="Replace THIS in text"> +<button id="button">Replace THIS</button> + +<script> +button.onclick = () => { + let pos = input.value.indexOf("THIS"); + if (pos >= 0) { + input.setRangeText("*THIS*", pos, pos + 4, "select"); + input.focus(); // focus to make selection visible + } +}; +</script> +``` + +### Example: insert at cursor + +If nothing is selected, or we use equal `start` and `end` in `setRangeText`, then the new text is just inserted, nothing is removed. + +We can also insert something "at the cursor" using `setRangeText`. + +Here's a button that inserts `"HELLO"` at the cursor position and puts the cursor immediately after it. If the selection is not empty, then it gets replaced (we can detect it by comparing `selectionStart!=selectionEnd` and do something else instead): + +```html run autorun +<input id="input" style="width:200px" value="Text Text Text Text Text"> +<button id="button">Insert "HELLO" at cursor</button> + +<script> + button.onclick = () => { + input.setRangeText("HELLO", input.selectionStart, input.selectionEnd, "end"); + input.focus(); + }; +</script> +``` + + +## Making unselectable + +To make something unselectable, there are three ways: + +1. Use CSS property `user-select: none`. + + ```html run + <style> + #elem { + user-select: none; + } + </style> + <div>Selectable <div id="elem">Unselectable</div> Selectable</div> + ``` + + This doesn't allow the selection to start at `elem`. But the user may start the selection elsewhere and include `elem` into it. + + Then `elem` will become a part of `document.getSelection()`, so the selection actually happens, but its content is usually ignored in copy-paste. + + +2. Prevent default action in `onselectstart` or `mousedown` events. + + ```html run + <div>Selectable <div id="elem">Unselectable</div> Selectable</div> + + <script> + elem.onselectstart = () => false; + </script> + ``` + + This prevents starting the selection on `elem`, but the visitor may start it at another element, then extend to `elem`. + + That's convenient when there's another event handler on the same action that triggers the select (e.g. `mousedown`). So we disable the selection to avoid conflict, still allowing `elem` contents to be copied. + +3. We can also clear the selection post-factum after it happens with `document.getSelection().empty()`. That's rarely used, as this causes unwanted blinking as the selection appears-disappears. + +## References + +- [DOM spec: Range](https://dom.spec.whatwg.org/#ranges) +- [Selection API](https://www.w3.org/TR/selection-api/#dom-globaleventhandlers-onselectstart) +- [HTML spec: APIs for the text control selections](https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#textFieldSelection) + + +## Summary + +We covered two different APIs for selections: + +1. For document: `Selection` and `Range` objects. +2. For `input`, `textarea`: additional methods and properties. + +The second API is very simple, as it works with text. + +The most used recipes are probably: + +1. Getting the selection: + ```js + let selection = document.getSelection(); + + let cloned = /* element to clone the selected nodes to */; + + // then apply Range methods to selection.getRangeAt(0) + // or, like here, to all ranges to support multi-select + for (let i = 0; i < selection.rangeCount; i++) { + cloned.append(selection.getRangeAt(i).cloneContents()); + } + ``` +2. Setting the selection: + ```js + let selection = document.getSelection(); + + // directly: + selection.setBaseAndExtent(...from...to...); + + // or we can create a range and: + selection.removeAllRanges(); + selection.addRange(range); + ``` + +And finally, about the cursor. The cursor position in editable elements, like `<textarea>` is always at the start or the end of the selection. We can use it to get cursor position or to move the cursor by setting `elem.selectionStart` and `elem.selectionEnd`. diff --git a/2-ui/99-ui-misc/02-selection-range/range-example-p-0-1.svg b/2-ui/99-ui-misc/02-selection-range/range-example-p-0-1.svg new file mode 100644 index 000000000..a97d1b47a --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/range-example-p-0-1.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="640" height="89" viewBox="0 0 640 89"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M47 35h262v21H47z"/><mask id="mask-2" width="262" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="range-example-p-0-1.svg"><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(13 35)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g><path id="Path-Copy" stroke="#C06334" stroke-width="2" d="M48 57v11h102V57"/><path id="Path-Copy-2" stroke="#C06334" stroke-width="2" d="M154 57v11h154V57"/><text id="0" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="84">0</tspan></text><text id="1" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="226" y="84">1</tspan></text><text id="2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="334" y="84">2</tspan></text><text id="3" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="428" y="84">3</tspan></text><path id="Path-Copy-3" stroke="#C06334" stroke-width="2" d="M312 57v11h53V57"/><path id="Path-Copy-4" stroke="#C06334" stroke-width="2" d="M369 57v11h128V57"/></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/range-example-p-1-3.svg b/2-ui/99-ui-misc/02-selection-range/range-example-p-1-3.svg new file mode 100644 index 000000000..2a8f9aca3 --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/range-example-p-1-3.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="640" height="89" viewBox="0 0 640 89"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M153 35h345v21H153z"/><mask id="mask-2" width="345" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="range-example-p-1-3.svg"><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><path id="Path-Copy" stroke="#C06334" stroke-width="2" d="M48 57v11h102V57"/><path id="Path-Copy-2" stroke="#C06334" stroke-width="2" d="M154 57v11h154V57"/><text id="0" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="84">0</tspan></text><text id="1" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="226" y="84">1</tspan></text><text id="2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="334" y="84">2</tspan></text><text id="3" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="428" y="84">3</tspan></text><path id="Path-Copy-3" stroke="#C06334" stroke-width="2" d="M312 57v11h53V57"/><path id="Path-Copy-4" stroke="#C06334" stroke-width="2" d="M369 57v11h128V57"/><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(13 35)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3-range.svg b/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3-range.svg new file mode 100644 index 000000000..32843436d --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3-range.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="683" height="229" viewBox="0 0 683 229"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M183 54h367v21H183z"/><mask id="mask-2" width="367" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="range-example-p-2-b-3-range.svg"><path id="Path" stroke="#C06334" stroke-width="2" d="M160 68v10h106V68"/><path id="Path-Copy-2" stroke="#C06334" stroke-width="2" d="M126 143v19h532v-19"/><path id="Path-Copy" stroke="#C06334" stroke-width="2" d="M514 70v10h50V70"/><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><text id="startContainer-(<p>." fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="10.3" y="118">startContainer</tspan> <tspan x=".7" y="136">(<p>.firstChild)</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M183 74l7 14h-6v16h-2V88h-6l7-14zM155 81l-6.166 14.387-3.876-4.581-15.312 12.957-.763.646-1.292-1.526.763-.646 15.312-12.958-3.875-4.58L155 81zM501 82l-6.166 14.387-3.876-4.581-15.312 12.957-.763.646-1.292-1.526.763-.646 15.312-12.958-3.875-4.58L501 82z"/><path id="Line-Copy-2" fill="#C06334" fill-rule="nonzero" d="M550 73l7 14h-6v16h-2V87h-6l7-14z"/><text id="startOffset-(=2)" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="170.2" y="118">startOffset</tspan> <tspan x="203.8" y="136">(=2)</tspan></text><text id="commonAncestorContai" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="316.1" y="181">commonAncestorContainer</tspan> <tspan x="402.5" y="199">(<p>)</tspan></text><text id="endContainer-(<b>.fi" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="388.9" y="117">endContainer</tspan> <tspan x="369.7" y="135">(<b>.firstChild)</tspan></text><text id="endOffset-(=3)" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="537.3" y="118">endOffset</tspan> <tspan x="561.3" y="136">(=3)</tspan></text><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(125 54)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3.svg b/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3.svg new file mode 100644 index 000000000..859f755ce --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/range-example-p-2-b-3.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="640" height="89" viewBox="0 0 640 89"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M71 35h367v21H71z"/><mask id="mask-2" width="367" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="range-example-p-2-b-3.svg"><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><path id="Path-Copy" stroke="#C06334" stroke-width="2" d="M48 57v11h102V57"/><path id="Path-Copy-2" stroke="#C06334" stroke-width="2" d="M154 57v11h154V57"/><text id="0" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="84">0</tspan></text><text id="1" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="226" y="84">1</tspan></text><text id="2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="334" y="84">2</tspan></text><text id="3" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="428" y="84">3</tspan></text><path id="Path-Copy-3" stroke="#C06334" stroke-width="2" d="M312 57v11h53V57"/><path id="Path-Copy-4" stroke="#C06334" stroke-width="2" d="M369 57v11h128V57"/><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(13 35)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/range-hello-1.svg b/2-ui/99-ui-misc/02-selection-range/range-hello-1.svg new file mode 100644 index 000000000..2951607a2 --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/range-hello-1.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="147" height="80" viewBox="0 0 147 80"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M63 17h25v21H63z"/><mask id="mask-2" width="25" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="range-hello-1.svg"><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><text id="<p>Hello</p>" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="18" font-weight="normal" letter-spacing="1"><tspan x="5" y="33"><p>Hello</p></tspan></text><path id="Path-Copy" stroke="#C06334" stroke-width="2" d="M41 38v11h58.887V37"/><text id="p.firstChild" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="25" y="69">p.firstChild</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/selection-direction-backward.svg b/2-ui/99-ui-misc/02-selection-range/selection-direction-backward.svg new file mode 100644 index 000000000..85615d38f --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/selection-direction-backward.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="559" height="94" viewBox="0 0 559 94"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M47 36h262v21H47z"/><mask id="mask-2" width="262" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="selection-direction-backward.svg"><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><text id="focus" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="25" y="80">focus</tspan></text><text id="anchor" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="277.2" y="80">anchor</tspan></text><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M62 21v6h247v2H62v6l-14-7 14-7z"/><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(13 37)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g><text id="mouse-move-direction" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="22">mouse move direction</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/selection-direction-forward.svg b/2-ui/99-ui-misc/02-selection-range/selection-direction-forward.svg new file mode 100644 index 000000000..511b00a26 --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/selection-direction-forward.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="563" height="94" viewBox="0 0 563 94"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="path-1" d="M47 38h262v21H47z"/><mask id="mask-2" width="262" height="21" x="0" y="0" fill="#fff" maskContentUnits="userSpaceOnUse" maskUnits="objectBoundingBox"><use xlink:href="#path-1"/></mask></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="selection-direction-forward.svg"><text id="anchor" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="18.2" y="78">anchor</tspan></text><text id="focus" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="285" y="78">focus</tspan></text><use id="Rectangle" fill="#C9DCEA" stroke="#C9DCEA" stroke-dasharray="5,2" stroke-width="2" mask="url(#mask-2)" xlink:href="#path-1"/><g id="<p>Example:-<i>itali" fill="#AF6E24" fill-rule="nonzero" transform="translate(13 39)"><path id="<p>Example:<i>italic</i>and<b>bold</b></p>" d="M.81 10.51v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08L.81 10.51zm12.718-.36a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM37.326 3.4h7.398v1.332H38.82v4.122h5.454v1.332H38.82v4.482h5.994V16h-7.488V3.4zm14.428 7.992L48.28 7h1.8l2.592 3.384L55.336 7h1.62l-3.474 4.32 3.69 4.68h-1.728l-2.862-3.69L49.666 16h-1.638l3.726-4.608zm9.19-3.69a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM75.516 16v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V16h-1.368V7h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V16h-1.368V9.88c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V16h-1.368zm8.812-5.85a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zM96.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm19.18 11.466a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm10.27-.036c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm20.216-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zM155.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM178.89 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zM189.358 7h2.142V5.218l1.404-.396V7h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611V8.206h-2.142V7zm13.186.702a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zM214.542 3.4h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zM226.09 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm15.166 3.942a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm4.708 2.124v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zM273.29 16v-1.206h3.132V8.206h-3.132V7h4.572v7.794h3.06V16h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm8.758 10.962l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm24.05-7.704a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V16h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm8.884-4.806a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.412l.162 1.458h.09c.12-.204.279-.405.477-.603a3.741 3.741 0 011.575-.927 3.593 3.593 0 011.026-.144c.552 0 1.041.06 1.467.18.426.12.78.333 1.062.639.282.306.495.72.639 1.242.144.522.216 1.185.216 1.989V16h-1.404v-4.896c0-.996-.162-1.746-.486-2.25-.324-.504-.912-.756-1.764-.756-.312 0-.615.063-.909.189a3.137 3.137 0 00-.801.495c-.24.204-.45.441-.63.711-.18.27-.312.555-.396.855V16h-1.386v-5.85zm16.408-6.75h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm19.208-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm11.026-7.11h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zM389.436 3.4h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.192 3.51c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm11.422-8.1h2.988v9.576c0 .732.123 1.248.369 1.548.246.3.609.45 1.089.45.336 0 .657-.06.963-.18.306-.12.645-.324 1.017-.612l.648.99a3.988 3.988 0 01-.63.45 4.678 4.678 0 01-1.422.54c-.24.048-.462.072-.666.072a3.81 3.81 0 01-1.188-.171 2.03 2.03 0 01-.873-.549c-.234-.252-.411-.585-.531-.999-.12-.414-.18-.927-.18-1.539v-8.37h-1.584V3.4zm15.94 0h2.916v9.522c0 .108.006.243.018.405a20.058 20.058 0 00.108 1.044c.024.174.048.327.072.459h1.206V16h-2.394l-.18-1.386h-.072c-.264.456-.657.837-1.179 1.143-.522.306-1.125.459-1.809.459-1.356 0-2.355-.381-2.997-1.143-.642-.762-.963-1.947-.963-3.555 0-.756.108-1.425.324-2.007a3.873 3.873 0 01.927-1.458c.402-.39.885-.687 1.449-.891.564-.204 1.194-.306 1.89-.306.252 0 .477.006.675.018.198.012.381.03.549.054.168.024.327.057.477.099.15.042.315.087.495.135V4.606h-1.512V3.4zm-1.08 11.61c.732 0 1.308-.189 1.728-.567.42-.378.708-.945.864-1.701V8.566a2.788 2.788 0 00-.891-.396c-.33-.084-.765-.126-1.305-.126-.96 0-1.716.279-2.268.837-.552.558-.828 1.437-.828 2.637 0 .492.045.951.135 1.377.09.426.24.795.45 1.107.21.312.489.558.837.738.348.18.774.27 1.278.27zm7.408-4.5v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm3.718.216h2.88v4.626h.09a3.082 3.082 0 011.206-.918 3.94 3.94 0 011.602-.324c2.64 0 3.96 1.524 3.96 4.572 0 1.548-.426 2.733-1.278 3.555-.852.822-2.064 1.233-3.636 1.233a8.655 8.655 0 01-2.016-.216c-.6-.144-1.038-.312-1.314-.504V4.606h-1.494V3.4zm5.598 4.59c-.72 0-1.302.204-1.746.612-.444.408-.768.984-.972 1.728v4.14c.276.144.606.258.99.342a5.63 5.63 0 001.206.126c.48 0 .915-.069 1.305-.207.39-.138.723-.354.999-.648.276-.294.489-.669.639-1.125.15-.456.225-.996.225-1.62 0-.456-.048-.888-.144-1.296a3.146 3.146 0 00-.459-1.071 2.208 2.208 0 00-.819-.72c-.336-.174-.744-.261-1.224-.261zm7.66 7.416l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896zm11.116-4.896v-.522l8.478-5.166.684 1.116-7.146 4.23 7.164 4.158-.684 1.08-8.496-4.896zm19.108-7.326l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm5.41 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V7h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm7.426 7.308l-.684-1.08 7.164-4.158-7.146-4.23.684-1.116 8.478 5.166v.522l-8.496 4.896z"/></g><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M296 21l14 7-14 7-.001-6H46v-2h249.999l.001-6z"/><text id="mouse-move-direction" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="93" y="22">mouse move direction</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/02-selection-range/selection-firefox.svg b/2-ui/99-ui-misc/02-selection-range/selection-firefox.svg new file mode 100644 index 000000000..aa7ff1eb7 --- /dev/null +++ b/2-ui/99-ui-misc/02-selection-range/selection-firefox.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="556" height="428" viewBox="0 0 556 428"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><defs><path id="rect-1" d="M48 88h448v281H48z"/></defs><g id="selection-range" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="selection-firefox.svg"><g id="Bitmap"><image width="448" height="281" x="48" y="88" xlink:href=""/><use stroke="#7E7C7B" xlink:href="#rect-1"/></g><path id="Line" fill="#C06334" fill-rule="nonzero" d="M309.315 59.871l1.064 1.694-.847.532-130.147 81.799 3.193 5.08L167 150.5l8.128-13.377 3.193 5.079 130.147-81.799.847-.532z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M349.496 60.872l.004 1 .45 128.874 6.001-.02L349 204.75l-7.049-13.975 5.999-.022-.45-128.875-.003-1 2-.006z"/><path id="Line-Copy-2" fill="#C06334" fill-rule="nonzero" d="M386.71 57.027l.256.966 35.404 133.219 5.8-1.54L425 205l-10.361-11.732 5.798-1.542-35.403-133.22-.257-.966 1.933-.513z"/><text id="selection" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="295" y="44">selection</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/solution.md b/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/solution.md new file mode 100644 index 000000000..2911b76cf --- /dev/null +++ b/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/solution.md @@ -0,0 +1,50 @@ +The console output is: 1 7 3 5 2 6 4. + +The task is quite simple, we just need to know how microtask and macrotask queues work. + +Let's see what's going on, step by step. + +```js +console.log(1); +// The first line executes immediately, it outputs `1`. +// Macrotask and microtask queues are empty, as of now. + +setTimeout(() => console.log(2)); +// `setTimeout` appends the callback to the macrotask queue. +// - macrotask queue content: +// `console.log(2)` + +Promise.resolve().then(() => console.log(3)); +// The callback is appended to the microtask queue. +// - microtask queue content: +// `console.log(3)` + +Promise.resolve().then(() => setTimeout(() => console.log(4))); +// The callback with `setTimeout(...4)` is appended to microtasks +// - microtask queue content: +// `console.log(3); setTimeout(...4)` + +Promise.resolve().then(() => console.log(5)); +// The callback is appended to the microtask queue +// - microtask queue content: +// `console.log(3); setTimeout(...4); console.log(5)` + +setTimeout(() => console.log(6)); +// `setTimeout` appends the callback to macrotasks +// - macrotask queue content: +// `console.log(2); console.log(6)` + +console.log(7); +// Outputs 7 immediately. +``` + +To summarize, + +1. Numbers `1` and `7` show up immediately, because simple `console.log` calls don't use any queues. +2. Then, after the main code flow is finished, the microtask queue runs. + - It has commands: `console.log(3); setTimeout(...4); console.log(5)`. + - Numbers `3` and `5` show up, while `setTimeout(() => console.log(4))` adds the `console.log(4)` call to the end of the macrotask queue. + - The macrotask queue is now: `console.log(2); console.log(6); console.log(4)`. +3. After the microtask queue becomes empty, the macrotask queue executes. It outputs `2`, `6`, `4`. + +Finally, we have the output: `1 7 3 5 2 6 4`. \ No newline at end of file diff --git a/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/task.md b/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/task.md new file mode 100644 index 000000000..ad406b3be --- /dev/null +++ b/2-ui/99-ui-misc/03-event-loop/2-micro-macro-queue/task.md @@ -0,0 +1,21 @@ +importance: 5 + +--- + +# What will be the output of this code? + +```js +console.log(1); + +setTimeout(() => console.log(2)); + +Promise.resolve().then(() => console.log(3)); + +Promise.resolve().then(() => setTimeout(() => console.log(4))); + +Promise.resolve().then(() => console.log(5)); + +setTimeout(() => console.log(6)); + +console.log(7); +``` diff --git a/2-ui/99-ui-misc/03-event-loop/article.md b/2-ui/99-ui-misc/03-event-loop/article.md new file mode 100644 index 000000000..f33188491 --- /dev/null +++ b/2-ui/99-ui-misc/03-event-loop/article.md @@ -0,0 +1,339 @@ + +# Event loop: microtasks and macrotasks + +Browser JavaScript execution flow, as well as in Node.js, is based on an *event loop*. + +Understanding how event loop works is important for optimizations, and sometimes for the right architecture. + +In this chapter we first cover theoretical details about how things work, and then see practical applications of that knowledge. + +## Event Loop + +The *event loop* concept is very simple. There's an endless loop, where the JavaScript engine waits for tasks, executes them and then sleeps, waiting for more tasks. + +The general algorithm of the engine: + +1. While there are tasks: + - execute them, starting with the oldest task. +2. Sleep until a task appears, then go to 1. + +That's a formalization of what we see when browsing a page. The JavaScript engine does nothing most of the time, it only runs if a script/handler/event activates. + +Examples of tasks: + +- When an external script `<script src="...">` loads, the task is to execute it. +- When a user moves their mouse, the task is to dispatch `mousemove` event and execute handlers. +- When the time is due for a scheduled `setTimeout`, the task is to run its callback. +- ...and so on. + +Tasks are set -- the engine handles them -- then waits for more tasks (while sleeping and consuming close to zero CPU). + +It may happen that a task comes while the engine is busy, then it's enqueued. + +The tasks form a queue, the so-called "macrotask queue" ([v8](https://v8.dev/) term): + + + +For instance, while the engine is busy executing a `script`, a user may move their mouse causing `mousemove`, and `setTimeout` may be due and so on, these tasks form a queue, as illustrated in the picture above. + +Tasks from the queue are processed on a "first come – first served" basis. When the engine browser is done with the `script`, it handles `mousemove` event, then `setTimeout` handler, and so on. + +So far, quite simple, right? + +Two more details: +1. Rendering never happens while the engine executes a task. It doesn't matter if the task takes a long time. Changes to the DOM are painted only after the task is complete. +2. If a task takes too long, the browser can't do other tasks, such as processing user events. So after some time, it raises an alert like "Page Unresponsive", suggesting killing the task with the whole page. That happens when there are a lot of complex calculations or a programming error leading to an infinite loop. + +That was the theory. Now let's see how we can apply that knowledge. + +## Use-case 1: splitting CPU-hungry tasks + +Let's say we have a CPU-hungry task. + +For example, syntax-highlighting (used to colorize code examples on this page) is quite CPU-heavy. To highlight the code, it performs the analysis, creates many colored elements, adds them to the document -- for a large amount of text that takes a lot of time. + +While the engine is busy with syntax highlighting, it can't do other DOM-related stuff, process user events, etc. It may even cause the browser to "hiccup" or even "hang" for a bit, which is unacceptable. + +We can avoid problems by splitting the big task into pieces. Highlight the first 100 lines, then schedule `setTimeout` (with zero-delay) for the next 100 lines, and so on. + +To demonstrate this approach, for the sake of simplicity, instead of text-highlighting, let's take a function that counts from `1` to `1000000000`. + +If you run the code below, the engine will "hang" for some time. For server-side JS that's clearly noticeable, and if you are running it in-browser, then try to click other buttons on the page -- you'll see that no other events get handled until the counting finishes. + +```js run +let i = 0; + +let start = Date.now(); + +function count() { + + // do a heavy job + for (let j = 0; j < 1e9; j++) { + i++; + } + + alert("Done in " + (Date.now() - start) + 'ms'); +} + +count(); +``` + +The browser may even show a "the script takes too long" warning. + +Let's split the job using nested `setTimeout` calls: + +```js run +let i = 0; + +let start = Date.now(); + +function count() { + + // do a piece of the heavy job (*) + do { + i++; + } while (i % 1e6 != 0); + + if (i == 1e9) { + alert("Done in " + (Date.now() - start) + 'ms'); + } else { + setTimeout(count); // schedule the new call (**) + } + +} + +count(); +``` + +Now the browser interface is fully functional during the "counting" process. + +A single run of `count` does a part of the job `(*)`, and then re-schedules itself `(**)` if needed: + +1. First run counts: `i=1...1000000`. +2. Second run counts: `i=1000001..2000000`. +3. ...and so on. + +Now, if a new side task (e.g. `onclick` event) appears while the engine is busy executing part 1, it gets queued and then executes when part 1 finished, before the next part. Periodic returns to the event loop between `count` executions provide just enough "air" for the JavaScript engine to do something else, to react to other user actions. + +The notable thing is that both variants -- with and without splitting the job by `setTimeout` -- are comparable in speed. There's not much difference in the overall counting time. + +To make them closer, let's make an improvement. + +We'll move the scheduling to the beginning of the `count()`: + +```js run +let i = 0; + +let start = Date.now(); + +function count() { + + // move the scheduling to the beginning + if (i < 1e9 - 1e6) { + setTimeout(count); // schedule the new call + } + + do { + i++; + } while (i % 1e6 != 0); + + if (i == 1e9) { + alert("Done in " + (Date.now() - start) + 'ms'); + } + +} + +count(); +``` + +Now when we start to `count()` and see that we'll need to `count()` more, we schedule that immediately, before doing the job. + +If you run it, it's easy to notice that it takes significantly less time. + +Why? + +That's simple: as you remember, there's the in-browser minimal delay of 4ms for many nested `setTimeout` calls. Even if we set `0`, it's `4ms` (or a bit more). So the earlier we schedule it - the faster it runs. + +Finally, we've split a CPU-hungry task into parts - now it doesn't block the user interface. And its overall execution time isn't much longer. + +## Use case 2: progress indication + +Another benefit of splitting heavy tasks for browser scripts is that we can show progress indication. + +As mentioned earlier, changes to DOM are painted only after the currently running task is completed, irrespective of how long it takes. + +On one hand, that's great, because our function may create many elements, add them one-by-one to the document and change their styles -- the visitor won't see any "intermediate", unfinished state. An important thing, right? + +Here's the demo, the changes to `i` won't show up until the function finishes, so we'll see only the last value: + + +```html run +<div id="progress"></div> + +<script> + + function count() { + for (let i = 0; i < 1e6; i++) { + i++; + progress.innerHTML = i; + } + } + + count(); +</script> +``` + +...But we also may want to show something during the task, e.g. a progress bar. + +If we split the heavy task into pieces using `setTimeout`, then changes are painted out in-between them. + +This looks prettier: + +```html run +<div id="progress"></div> + +<script> + let i = 0; + + function count() { + + // do a piece of the heavy job (*) + do { + i++; + progress.innerHTML = i; + } while (i % 1e3 != 0); + + if (i < 1e7) { + setTimeout(count); + } + + } + + count(); +</script> +``` + +Now the `<div>` shows increasing values of `i`, a kind of a progress bar. + + +## Use case 3: doing something after the event + +In an event handler we may decide to postpone some actions until the event bubbled up and was handled on all levels. We can do that by wrapping the code in zero delay `setTimeout`. + +In the chapter <info:dispatch-events> we saw an example: custom event `menu-open` is dispatched in `setTimeout`, so that it happens after the "click" event is fully handled. + +```js +menu.onclick = function() { + // ... + + // create a custom event with the clicked menu item data + let customEvent = new CustomEvent("menu-open", { + bubbles: true + }); + + // dispatch the custom event asynchronously + setTimeout(() => menu.dispatchEvent(customEvent)); +}; +``` + +## Macrotasks and Microtasks + +Along with *macrotasks*, described in this chapter, there are *microtasks*, mentioned in the chapter <info:microtask-queue>. + +Microtasks come solely from our code. They are usually created by promises: an execution of `.then/catch/finally` handler becomes a microtask. Microtasks are used "under the cover" of `await` as well, as it's another form of promise handling. + +There's also a special function `queueMicrotask(func)` that queues `func` for execution in the microtask queue. + +**Immediately after every *macrotask*, the engine executes all tasks from *microtask* queue, prior to running any other macrotasks or rendering or anything else.** + +For instance, take a look: + +```js run +setTimeout(() => alert("timeout")); + +Promise.resolve() + .then(() => alert("promise")); + +alert("code"); +``` + +What's going to be the order here? + +1. `code` shows first, because it's a regular synchronous call. +2. `promise` shows second, because `.then` passes through the microtask queue, and runs after the current code. +3. `timeout` shows last, because it's a macrotask. + +The richer event loop picture looks like this (order is from top to bottom, that is: the script first, then microtasks, rendering and so on): + + + +All microtasks are completed before any other event handling or rendering or any other macrotask takes place. + +That's important, as it guarantees that the application environment is basically the same (no mouse coordinate changes, no new network data, etc) between microtasks. + +If we'd like to execute a function asynchronously (after the current code), but before changes are rendered or new events handled, we can schedule it with `queueMicrotask`. + +Here's an example with "counting progress bar", similar to the one shown previously, but `queueMicrotask` is used instead of `setTimeout`. You can see that it renders at the very end. Just like the synchronous code: + +```html run +<div id="progress"></div> + +<script> + let i = 0; + + function count() { + + // do a piece of the heavy job (*) + do { + i++; + progress.innerHTML = i; + } while (i % 1e3 != 0); + + if (i < 1e6) { + *!* + queueMicrotask(count); + */!* + } + + } + + count(); +</script> +``` + +## Summary + +A more detailed event loop algorithm (though still simplified compared to the [specification](https://html.spec.whatwg.org/multipage/webappapis.html#event-loop-processing-model)): + +1. Dequeue and run the oldest task from the *macrotask* queue (e.g. "script"). +2. Execute all *microtasks*: + - While the microtask queue is not empty: + - Dequeue and run the oldest microtask. +3. Render changes if any. +4. If the macrotask queue is empty, wait till a macrotask appears. +5. Go to step 1. + +To schedule a new *macrotask*: +- Use zero delayed `setTimeout(f)`. + +That may be used to split a big calculation-heavy task into pieces, for the browser to be able to react to user events and show progress between them. + +Also, used in event handlers to schedule an action after the event is fully handled (bubbling done). + +To schedule a new *microtask* +- Use `queueMicrotask(f)`. +- Also promise handlers go through the microtask queue. + +There's no UI or network event handling between microtasks: they run immediately one after another. + +So one may want to `queueMicrotask` to execute a function asynchronously, but within the environment state. + +```smart header="Web Workers" +For long heavy calculations that shouldn't block the event loop, we can use [Web Workers](https://html.spec.whatwg.org/multipage/workers.html). + +That's a way to run code in another, parallel thread. + +Web Workers can exchange messages with the main process, but they have their own variables, and their own event loop. + +Web Workers do not have access to DOM, so they are useful, mainly, for calculations, to use multiple CPU cores simultaneously. +``` diff --git a/2-ui/99-ui-misc/03-event-loop/eventLoop-full.svg b/2-ui/99-ui-misc/03-event-loop/eventLoop-full.svg new file mode 100644 index 000000000..593cbab9b --- /dev/null +++ b/2-ui/99-ui-misc/03-event-loop/eventLoop-full.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="407" height="391" viewBox="0 0 407 391"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="eventLoop-full.svg"><text id="..." fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="257.4" y="71">...</tspan></text><path id="Rectangle-1-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M216 86h108v28H216z"/><path id="Rectangle-1-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M216 195h108v28H216z"/><text id="mousemove" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="232.2" y="214">mousemove</tspan></text><path id="Rectangle-1-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M216 302h108v28H216z"/><text id="event-loop" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="44.922" y="189">event</tspan> <tspan x="51.115" y="222">loop</tspan></text><text id="render-copy-2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="297.3" y="176">render</tspan></text><g id="np_paint_2296353_000000-copy-2" fill="#C06334" fill-rule="nonzero" transform="translate(256 158)"><path id="Shape" d="M3.079 0A3.095 3.095 0 000 3.079V22.92A3.095 3.095 0 003.079 26H22.92A3.095 3.095 0 0026 22.921V3.08A3.095 3.095 0 0022.921 0H3.08zm0 2.053H22.92c.587 0 1.026.44 1.026 1.026v13.043l-2.886-2.684a1.027 1.027 0 00-1.443.032l-5.72 5.955-5.035-4.148a1.021 1.021 0 00-1.38.075l-5.43 5.43V3.08c0-.587.44-1.026 1.026-1.026zM6.158 3.42A3.095 3.095 0 003.078 6.5a3.095 3.095 0 003.08 3.079A3.095 3.095 0 009.237 6.5a3.095 3.095 0 00-3.08-3.079zm0 2.053c.579 0 1.026.447 1.026 1.026s-.447 1.026-1.026 1.026A1.012 1.012 0 015.132 6.5c0-.579.447-1.026 1.026-1.026zm14.176.566c-.377 0-.759.722-.759 1.08-.46.112-.748.358-.748.716s.371.717.748.717h2.662c.73 0 1.11-1.433-.021-1.433-.022-.677-.445-.86-1.134-.716-.183-.284-.375-.386-.748-.364zM16.71 9.237c-.314 0-.631.584-.631.876-.382.093-.63.296-.63.588 0 .293.316.589.63.589h2.213c.608 0 .931-1.176-.01-1.176-.018-.555-.379-.706-.952-.588-.152-.232-.31-.308-.62-.29zm3.688 6.382l3.55 3.293v4.01c0 .586-.44 1.025-1.027 1.025H3.08a1.01 1.01 0 01-.855-.449l6.05-6.04 4.202 3.453-1.24 1.294 1.475 1.422 7.687-8.008z"/></g><text id="microtasks-copy-2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="297.5" y="140">microtasks</tspan></text><g id="Group-Copy-2" fill="#C06334" transform="translate(256 124)"><path id="Fill-52" d="M23.145.978C21.737.348 20.186 0 18.56 0c-2.59 0-4.983.885-6.921 2.38L10.474.618a.286.286 0 00-.305-.125.303.303 0 00-.226.249L8.797 7.779a.317.317 0 00.07.258.287.287 0 00.22.102h.018l3.112-.206 3.504-.233a.296.296 0 00.258-.204.315.315 0 00-.078-.33l-1.46-1.415a7.18 7.18 0 014.118-1.301c1.07 0 2.083.238 3.003.662a.284.284 0 00.23.003.306.306 0 00.162-.171l1.35-3.568a.314.314 0 00-.159-.398" transform="matrix(-1 0 0 1 32.117 0)"/><path id="Fill-54" d="M12.787 13.315a12.66 12.66 0 001.346 4.7c1.225 2.393 3.101 4.186 5.274 5.27l-.929 1.908a.318.318 0 00.04.342c.077.094.201.13.315.09l6.454-2.269a.3.3 0 00.183-.187.32.32 0 00-.018-.251l-.01-.016-1.644-2.779-1.852-3.127a.29.29 0 00-.295-.142.298.298 0 00-.24.228l-.498 2.018a7.607 7.607 0 01-3.041-3.19 8.083 8.083 0 01-.865-3.087.314.314 0 00-.105-.214.286.286 0 00-.22-.068l-3.636.44a.305.305 0 00-.26.334" transform="matrix(-1 0 0 1 38.273 0)"/><path id="Fill-56" d="M2.646 22.333a12.003 12.003 0 003.205-3.576 12.684 12.684 0 001.713-7.424l2.04-.112a.293.293 0 00.262-.207.316.316 0 00-.084-.332L4.681 5.958a.28.28 0 00-.247-.072.291.291 0 00-.198.14l-.01.018-1.47 2.883-1.654 3.247a.32.32 0 00.03.339.285.285 0 00.307.104l1.916-.558a8.078 8.078 0 01-1.111 4.358 7.652 7.652 0 01-2.117 2.33.31.31 0 00-.123.203.325.325 0 00.053.234l2.183 3.08a.285.285 0 00.406.069" transform="matrix(-1 0 0 1 9.882 0)"/></g><text id="render-copy-2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="297.3" y="285">render</tspan></text><g id="np_paint_2296353_000000-copy-2" fill="#C06334" fill-rule="nonzero" transform="translate(256 267)"><path id="Shape" d="M3.079 0A3.095 3.095 0 000 3.079V22.92A3.095 3.095 0 003.079 26H22.92A3.095 3.095 0 0026 22.921V3.08A3.095 3.095 0 0022.921 0H3.08zm0 2.053H22.92c.587 0 1.026.44 1.026 1.026v13.043l-2.886-2.684a1.027 1.027 0 00-1.443.032l-5.72 5.955-5.035-4.148a1.021 1.021 0 00-1.38.075l-5.43 5.43V3.08c0-.587.44-1.026 1.026-1.026zM6.158 3.42A3.095 3.095 0 003.078 6.5a3.095 3.095 0 003.08 3.079A3.095 3.095 0 009.237 6.5a3.095 3.095 0 00-3.08-3.079zm0 2.053c.579 0 1.026.447 1.026 1.026s-.447 1.026-1.026 1.026A1.012 1.012 0 015.132 6.5c0-.579.447-1.026 1.026-1.026zm14.176.566c-.377 0-.759.722-.759 1.08-.46.112-.748.358-.748.716s.371.717.748.717h2.662c.73 0 1.11-1.433-.021-1.433-.022-.677-.445-.86-1.134-.716-.183-.284-.375-.386-.748-.364zM16.71 9.237c-.314 0-.631.584-.631.876-.382.093-.63.296-.63.588 0 .293.316.589.63.589h2.213c.608 0 .931-1.176-.01-1.176-.018-.555-.379-.706-.952-.588-.152-.232-.31-.308-.62-.29zm3.688 6.382l3.55 3.293v4.01c0 .586-.44 1.025-1.027 1.025H3.08a1.01 1.01 0 01-.855-.449l6.05-6.04 4.202 3.453-1.24 1.294 1.475 1.422 7.687-8.008z"/></g><text id="microtasks-copy-2" fill="#181717" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="297.5" y="249">microtasks</tspan></text><g id="Group-Copy-2" fill="#C06334" transform="translate(256 233)"><path id="Fill-52" d="M23.145.978C21.737.348 20.186 0 18.56 0c-2.59 0-4.983.885-6.921 2.38L10.474.618a.286.286 0 00-.305-.125.303.303 0 00-.226.249L8.797 7.779a.317.317 0 00.07.258.287.287 0 00.22.102h.018l3.112-.206 3.504-.233a.296.296 0 00.258-.204.315.315 0 00-.078-.33l-1.46-1.415a7.18 7.18 0 014.118-1.301c1.07 0 2.083.238 3.003.662a.284.284 0 00.23.003.306.306 0 00.162-.171l1.35-3.568a.314.314 0 00-.159-.398" transform="matrix(-1 0 0 1 32.117 0)"/><path id="Fill-54" d="M12.787 13.315a12.66 12.66 0 001.346 4.7c1.225 2.393 3.101 4.186 5.274 5.27l-.929 1.908a.318.318 0 00.04.342c.077.094.201.13.315.09l6.454-2.269a.3.3 0 00.183-.187.32.32 0 00-.018-.251l-.01-.016-1.644-2.779-1.852-3.127a.29.29 0 00-.295-.142.298.298 0 00-.24.228l-.498 2.018a7.607 7.607 0 01-3.041-3.19 8.083 8.083 0 01-.865-3.087.314.314 0 00-.105-.214.286.286 0 00-.22-.068l-3.636.44a.305.305 0 00-.26.334" transform="matrix(-1 0 0 1 38.273 0)"/><path id="Fill-56" d="M2.646 22.333a12.003 12.003 0 003.205-3.576 12.684 12.684 0 001.713-7.424l2.04-.112a.293.293 0 00.262-.207.316.316 0 00-.084-.332L4.681 5.958a.28.28 0 00-.247-.072.291.291 0 00-.198.14l-.01.018-1.47 2.883-1.654 3.247a.32.32 0 00.03.339.285.285 0 00.307.104l1.916-.558a8.078 8.078 0 01-1.111 4.358 7.652 7.652 0 01-2.117 2.33.31.31 0 00-.123.203.325.325 0 00.053.234l2.183 3.08a.285.285 0 00.406.069" transform="matrix(-1 0 0 1 9.882 0)"/></g><path id="Path" fill="#C06334" fill-rule="nonzero" d="M240.536 23.5c18.067 0 28.19 5.407 30.493 19.865l.067.441.016.179.264 10.223 5.906-11.165 2.652 1.402-8.582 16.229-1.247 2.357-1.367-2.29-9.412-15.76 2.576-1.539 6.476 10.845-.263-10.128-.046-.305c-1.998-12.562-10.586-17.259-27.028-17.353l-.505-.001h-51.072c-18.854 0-27.785 5.925-27.961 22.709l-.003.512v294.675c0 9.183 2.661 14.683 7.881 17.703 4.3 2.487 10.047 3.365 19.488 3.4l.595.001h51.072c9.796 0 15.693-.862 20.083-3.401 5.127-2.966 7.785-8.324 7.878-17.214l.003-.489v-6.472h3v6.472c0 10.183-3.152 16.697-9.379 20.3-4.884 2.825-11.037 3.766-20.972 3.803l-.613.001h-51.072c-10.296 0-16.602-.921-21.585-3.804-6.122-3.542-9.271-9.897-9.376-19.785l-.003-.515V49.721c0-18.944 10.246-26.078 30.352-26.219l.612-.002h51.072z"/><text id="script" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="245.3" y="105">script</tspan></text><text id="setTimeout" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="228.5" y="321">setTimeout</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/03-event-loop/eventLoop.svg b/2-ui/99-ui-misc/03-event-loop/eventLoop.svg new file mode 100644 index 000000000..6dc459ef8 --- /dev/null +++ b/2-ui/99-ui-misc/03-event-loop/eventLoop.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="500" height="279" viewBox="0 0 500 279"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="promise" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="eventLoop.svg"><path id="Rectangle-1-Copy-5" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M218 173h108v28H218z"/><text id="..." fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="258.9" y="189">...</tspan></text><path id="Rectangle-1-Copy-6" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M218 90h108v28H218z"/><path id="Rectangle-1-Copy-8" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M218 117h108v28H218z"/><text id="mousemove" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="233.7" y="136">mousemove</tspan></text><path id="Rectangle-1-Copy-9" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M218 145h108v28H218z"/><text id="script" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="246.8" y="109">script</tspan></text><text id="event-loop" fill="#C06334" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="76.422" y="134">event</tspan> <tspan x="82.615" y="167">loop</tspan></text><text id="macrotask-queue" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="348.371" y="136">macrotask</tspan> <tspan x="371.451" y="169">queue</tspan></text><path id="Path" fill="#C06334" fill-rule="nonzero" d="M192 40c22.467 0 43.818 7.359 61.285 20.712l.622.479.143.122 8.961 8.422-4.584-14.61 3.817-1.197 6.869 21.895 1.064 3.392-3.452-.852-22.278-5.5.959-3.883 14.864 3.669-8.886-8.352-.55-.424c-16.564-12.656-36.757-19.698-58.036-19.87L192 44c-30.748 0-59.095 14.403-77.315 38.412l-.549.73-3.208-2.387C129.87 55.303 159.664 40 192 40z"/><path id="Path-Copy-2" fill="#C06334" fill-rule="nonzero" d="M269.882 208.148l2.823 2.834-.366.365a533.245 533.245 0 00-3.982 4.033l-.333.34c-9.922 10.12-14.79 14.544-22.017 19.272-15.185 9.934-34.01 15.688-51.594 15.688-25.222 0-47.144-6.827-64.077-19.673l-.589-.45-.089-.07-.08-.078-8.854-8.581 4.36 14.68-3.835 1.138-6.532-21.998-1.012-3.409 3.439.906 22.19 5.841-1.018 3.868-14.808-3.898 8.766 8.497.488.374c16.02 12.15 36.782 18.698 60.792 18.85l.859.003c16.796 0 34.862-5.522 49.404-15.035 6.911-4.521 11.624-8.806 21.35-18.725l.334-.34a633.126 633.126 0 013.46-3.511l.736-.736.185-.185z"/><text id="setTimeout" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="230" y="164">setTimeout</tspan></text></g></g></svg> \ No newline at end of file diff --git a/2-ui/99-ui-misc/index.md b/2-ui/99-ui-misc/index.md new file mode 100644 index 000000000..79cd72fe7 --- /dev/null +++ b/2-ui/99-ui-misc/index.md @@ -0,0 +1,2 @@ + +# Miscellaneous diff --git a/3-frames-and-windows/01-popup-windows/article.md b/3-frames-and-windows/01-popup-windows/article.md index 57c0c50d3..f2c87d1e0 100644 --- a/3-frames-and-windows/01-popup-windows/article.md +++ b/3-frames-and-windows/01-popup-windows/article.md @@ -7,13 +7,21 @@ Basically, you just run: window.open('https://javascript.info/') ``` -... And it will open a new window with given URL. Most modern browsers are configured to open new tabs instead of separate windows. +...And it will open a new window with given URL. Most modern browsers are configured to open url in new tabs instead of separate windows. -## Popup blocking +Popups exist from really ancient times. The initial idea was to show another content without closing the main window. As of now, there are other ways to do that: we can load content dynamically with [fetch](info:fetch) and show it in a dynamically generated `<div>`. So, popups is not something we use everyday. + +Also, popups are tricky on mobile devices, that don't show multiple windows simultaneously. + +Still, there are tasks where popups are still used, e.g. for OAuth authorization (login with Google/Facebook/...), because: + +1. A popup is a separate window which has its own independent JavaScript environment. So opening a popup from a third-party, non-trusted site is safe. +2. It's very easy to open a popup. +3. A popup can navigate (change URL) and send messages to the opener window. -Popups exist from really ancient times. The initial idea was to show another content without closing the main window. As of now, there are other ways to do that: JavaScript is able to send requests for server, so popups are rarely used. But sometimes they are still handy. +## Popup blocking -In the past evil sites abused popups a lot. A bad page could open tons of popup windows with ads. So now most browsers try to block popups and protect the user. +In the past, evil sites abused popups a lot. A bad page could open tons of popup windows with ads. So now most browsers try to block popups and protect the user. **Most browsers block popups if they are called outside of user-triggered event handlers like `onclick`.** @@ -30,38 +38,6 @@ button.onclick = () => { This way users are somewhat protected from unwanted popups, but the functionality is not disabled totally. -What if the popup opens from `onclick`, but after `setTimeout`? That's a bit tricky. - -Try this code: - -```js run -// open after 3 seconds -setTimeout(() => window.open('http://google.com'), 3000); -``` - -The popup opens in Chrome, but gets blocked in Firefox. - -...If we decrease the delay, the popup works in Firefox too: - -```js run -// open after 1 seconds -setTimeout(() => window.open('http://google.com'), 1000); -``` - -The difference is that Firefox treats a timeout of 2000ms or less are acceptable, but after it -- removes the "trust", assuming that now it's "outside of the user action". So the first one is blocked, and the second one is not. - -## Modern usage - -As of now, we have many methods to load and show data on-page with JavaScript. But there are still situations when a popup works best. - -For instance, many shops use online chats for consulting people. A visitor clicks on the button, it runs `window.open` and opens the popup with the chat. - -Why a popup is good here, why not in-page? - -1. A popup is a separate window with its own independent JavaScript environment. So a chat service doesn't need to integrate with scripts of the main shop site. -2. A popup is very simple to attach to a site, little to no overhead. It's only a small button, without additional scripts. -3. A popup may persist even if the user left the page. For example, a consult advices the user to visit the page of a new "Super-Cooler" goodie. The user goes there in the main window without leaving the chat. - ## window.open The syntax to open a popup is: `window.open(url, name, params)`: @@ -73,7 +49,7 @@ name : A name of the new window. Each window has a `window.name`, and here we can specify which window to use for the popup. If there's already a window with such name -- the given URL opens in it, otherwise a new window is opened. params -: The configuration string for the new window. It contains settings, delimited by a comma. There must be no spaces in params, for instance: `width:200,height=100`. +: The configuration string for the new window. It contains settings, delimited by a comma. There must be no spaces in params, for instance: `width=200,height=100`. Settings for `params`: @@ -91,9 +67,9 @@ Settings for `params`: There is also a number of less supported browser-specific features, which are usually not used. Check <a href="https://developer.mozilla.org/en/DOM/window.open">window.open in MDN</a> for examples. -## Example: a minimalistic window +## Example: a minimalistic window -Let's open a window with minimal set of features just to see which of them browser allows to disable: +Let's open a window with minimal set of features, just to see which of them browser allows to disable: ```js run let params = `scrollbars=no,resizable=no,status=no,location=no,toolbar=no,menubar=no, @@ -118,20 +94,30 @@ Most browsers show the example above as required. Rules for omitted settings: - If there is no 3rd argument in the `open` call, or it is empty, then the default window parameters are used. -- If there is a string of params, but some yes/no features are omitted, then the omitted features are disabled, if the browser allows that. So if you specify params, make sure you explicitly set all required features to yes. +- If there is a string of params, but some `yes/no` features are omitted, then the omitted features assumed to have `no` value. So if you specify params, make sure you explicitly set all required features to yes. - If there is no `left/top` in params, then the browser tries to open a new window near the last opened window. - If there is no `width/height`, then the new window will be the same size as the last opened. -## Accessing a popup +## Accessing popup from window + +The `open` call returns a reference to the new window. It can be used to manipulate its properties, change location and even more. -The `open` call returns a reference to the new window. It can be used to manipulate it's properties, change location and even more. +In this example, we generate popup content from JavaScript: -In the example below, the contents of the new window is modified after loading. +```js +let newWin = window.open("about:blank", "hello", "width=200,height=200"); + +newWin.document.write("Hello, world!"); +``` + +And here we modify the contents after loading: ```js run let newWindow = open('/', 'example', 'width=300,height=300') newWindow.focus(); +alert(newWindow.location.href); // (*) about:blank, loading hasn't started yet + newWindow.onload = function() { let html = `<div style="font-size:30px">Welcome!</div>`; *!* @@ -140,65 +126,133 @@ newWindow.onload = function() { }; ``` -Please note that external `document` content is only accessible for windows from the same origin (the same protocol://domain:port). +Please note: immediately after `window.open`, the new window isn't loaded yet. That's demonstrated by `alert` in line `(*)`. So we wait for `onload` to modify it. We could also use `DOMContentLoaded` handler for `newWin.document`. + +```warn header="Same origin policy" +Windows may freely access content of each other only if they come from the same origin (the same protocol://domain:port). + +Otherwise, e.g. if the main window is from `site.com`, and the popup from `gmail.com`, that's impossible for user safety reasons. For the details, see chapter <info:cross-window-communication>. +``` + +## Accessing window from popup -For windows with URLs from another sites, we are able to change the location by assigning `newWindow.location=...`, but we can't read the location or access the content. That's for user safety, so that an evil page can't open a popup with `http://gmail.com` and read the data. We'll talk more about it later. +A popup may access the "opener" window as well using `window.opener` reference. It is `null` for all windows except popups. -## Accessing the opener window +If you run the code below, it replaces the opener (current) window content with "Test": -A popup may access the "opener" window as well. A JavaScript in it may use `window.opener` to access the window that opened it. It is `null` for all windows except popups. +```js run +let newWin = window.open("about:blank", "hello", "width=200,height=200"); + +newWin.document.write( + "<script>window.opener.document.body.innerHTML = 'Test'<\/script>" +); +``` -So both the main window and the popup have a reference to each other. They may modify each other freely assuming that they come from the same origin. If that's not so, then there are still means to communicate, to be covered in the next chapter <info:cross-window-communication>. +So the connection between the windows is bidirectional: the main window and the popup have a reference to each other. ## Closing a popup -If we don't need a popup any more, we can call `newWindow.close()` on it. +To close a window: `win.close()`. + +To check if a window is closed: `win.closed`. -Technically, the `close()` method is available for any `window`, but `window.close()` is ignored by most browsers if `window` is not created with `window.open()`. +Technically, the `close()` method is available for any `window`, but `window.close()` is ignored by most browsers if `window` is not created with `window.open()`. So it'll only work on a popup. -The `newWindow.closed` is `true` if the window is closed. That's useful to check if the popup (or the main window) is still open or not. A user could close it, and our code should take that possibility into account. +The `closed` property is `true` if the window is closed. That's useful to check if the popup (or the main window) is still open or not. A user can close it anytime, and our code should take that possibility into account. This code loads and then closes the window: ```js run -let newWindow = open('/', 'example', 'width=300,height=300') +let newWindow = open('/', 'example', 'width=300,height=300'); + newWindow.onload = function() { newWindow.close(); alert(newWindow.closed); // true }; ``` -## Focus/blur on a popup -Theoretically, there are `window.focus()` and `window.blur()` methods to focus/unfocus on a window. Also there are `focus/blur` events that allow to focus a window and catch the moment when the visitor switches elsewhere. +## Moving and resizing + +There are methods to move/resize a window: + +`win.moveBy(x,y)` +: Move the window relative to current position `x` pixels to the right and `y` pixels down. Negative values are allowed (to move left/up). -In the past evil pages abused those. For instance, look at this code: +`win.moveTo(x,y)` +: Move the window to coordinates `(x,y)` on the screen. + +`win.resizeBy(width,height)` +: Resize the window by given `width/height` relative to the current size. Negative values are allowed. + +`win.resizeTo(width,height)` +: Resize the window to the given size. + +There's also `window.onresize` event. + +```warn header="Only popups" +To prevent abuse, the browser usually blocks these methods. They only work reliably on popups that we opened, that have no additional tabs. +``` + +```warn header="No minification/maximization" +JavaScript has no way to minify or maximize a window. These OS-level functions are hidden from Frontend-developers. + +Move/resize methods do not work for maximized/minimized windows. +``` + +## Scrolling a window + +We already talked about scrolling a window in the chapter <info:size-and-scroll-window>. + +`win.scrollBy(x,y)` +: Scroll the window `x` pixels right and `y` down relative the current scroll. Negative values are allowed. + +`win.scrollTo(x,y)` +: Scroll the window to the given coordinates `(x,y)`. + +`elem.scrollIntoView(top = true)` +: Scroll the window to make `elem` show up at the top (the default) or at the bottom for `elem.scrollIntoView(false)`. + +There's also `window.onscroll` event. + +## Focus/blur on a window + +Theoretically, there are `window.focus()` and `window.blur()` methods to focus/unfocus on a window. And there are also `focus/blur` events that allow to catch the moment when the visitor focuses on a window and switches elsewhere. + +Although, in practice they are severely limited, because in the past evil pages abused them. + +For instance, look at this code: ```js run window.onblur = () => window.focus(); ``` -When a user attempts to switch out of the window (`blur`), it brings it back to focus. The intention is to "lock" the user within the `window`. +When a user attempts to switch out of the window (`window.onblur`), it brings the window back into focus. The intention is to "lock" the user within the `window`. -So, there are limitations that forbid the code like that. There are many limitations to protect the user from ads and evils pages. They depend on the browser. +So browsers had to introduce many limitations to forbid the code like that and protect the user from ads and evils pages. They depend on the browser. -For instance, a mobile browser usually ignores that call completely. Also focusing doesn't work when a popup opens in a separate tab rather than a new window. +For instance, a mobile browser usually ignores `window.focus()` completely. Also focusing doesn't work when a popup opens in a separate tab rather than a new window. -Still, there are some things that can be done. +Still, there are some use cases when such calls do work and can be useful. For instance: -- When we open a popup, it's might be a good idea to run a `newWindow.focus()` on it. Just in case, for some OS/browser combinations it ensures that the user is in the new window now. +- When we open a popup, it might be a good idea to run `newWindow.focus()` on it. Just in case, for some OS/browser combinations it ensures that the user is in the new window now. - If we want to track when a visitor actually uses our web-app, we can track `window.onfocus/onblur`. That allows us to suspend/resume in-page activities, animations etc. But please note that the `blur` event means that the visitor switched out from the window, but they still may observe it. The window is in the background, but still may be visible. -## Summary +## Summary + +Popup windows are used rarely, as there are alternatives: loading and displaying information in-page, or in iframe. + +If we're going to open a popup, a good practice is to inform the user about it. An "opening window" icon near a link or button would allow the visitor to survive the focus shift and keep both windows in mind. - A popup can be opened by the `open(url, name, params)` call. It returns the reference to the newly opened window. -- By default, browsers block `open` calls from the code outside of user actions. Usually a notification appears, so that a user may allow them. -- The popup may access the opener window using the `window.opener` property, so the two are connected. -- If the main window and the popup come from the same origin, they can freely read and modify each other. Otherwise, they can change location of each other and communicate using messages (to be covered). -- To close the popup: use `close()` call. Also the user may close them (just like any other windows). The `window.closed` is `true` after that. -- Methods `focus()` and `blur()` allow to focus/unfocus a window. Sometimes. -- Events `focus` and `blur` allow to track switching in and out of the window. But please note that a window may still be visible even in the background state, after `blur`. +- Browsers block `open` calls from the code outside of user actions. Usually a notification appears, so that a user may allow them. +- Browsers open a new tab by default, but if sizes are provided, then it'll be a popup window. +- The popup may access the opener window using the `window.opener` property. +- The main window and the popup can freely read and modify each other if they have the same origin. Otherwise, they can change location of each other and [exchange messages](info:cross-window-communication). + +To close the popup: use `close()` call. Also the user may close them (just like any other windows). The `window.closed` is `true` after that. -Also if we open a popup, a good practice is to notify the user about it. An icon with the opening window can help the visitor to survive the focus shift and keep both windows in mind. +- Methods `focus()` and `blur()` allow to focus/unfocus a window. But they don't work all the time. +- Events `focus` and `blur` allow to track switching in and out of the window. But please note that a window may still be visible even in the background state, after `blur`. diff --git a/3-frames-and-windows/03-cross-window-communication/article.md b/3-frames-and-windows/03-cross-window-communication/article.md index 53cb3cd6e..4d4e320e4 100644 --- a/3-frames-and-windows/03-cross-window-communication/article.md +++ b/3-frames-and-windows/03-cross-window-communication/article.md @@ -26,37 +26,18 @@ The "Same Origin" policy states that: - if we have a reference to another window, e.g. a popup created by `window.open` or a window inside `<iframe>`, and that window comes from the same origin, then we have full access to that window. - otherwise, if it comes from another origin, then we can't access the content of that window: variables, document, anything. The only exception is `location`: we can change it (thus redirecting the user). But we cannot *read* location (so we can't see where the user is now, no information leak). -Now let's see some examples. First, we'll look at pages that come from the same origin and do not conflict with the "Same Origin" policy, and afterwards we'll cover cross-window messaging that allows to work around the "Same Origin" policy. +### In action: iframe +An `<iframe>` tag hosts a separate embedded window, with its own separate `document` and `window` objects. -````warn header="Subdomains may be same-origin" -There's a small exclusion in the "Same Origin" policy. +We can access them using properties: -If windows share the same second-level domain, for instance `john.site.com`, `peter.site.com` and `site.com` (so that their common second-level domain is `site.com`), they can be treated as coming from the "same origin". +- `iframe.contentWindow` to get the window inside the `<iframe>`. +- `iframe.contentDocument` to get the document inside the `<iframe>`, shorthand for `iframe.contentWindow.document`. -To make it work, all such pages (including the one from `site.com`) should run the code: +When we access something inside the embedded window, the browser checks if the iframe has the same origin. If that's not so then the access is denied (writing to `location` is an exception, it's still permitted). -```js -document.domain = 'site.com'; -``` - -That's all. Now they can interact without limitations. Again, that's only possible for pages with the same second-level domain. -```` - -## Accessing an iframe contents - -Our first example covers iframes. An `<iframe>` is a two-faced beast. From one side it's a tag, just like `<script>` or `<img>`. From the other side it's a window-in-window. - -The embedded window has a separate `document` and `window` objects. - -We can access them like using the properties: - -- `iframe.contentWindow` is a reference to the window inside the `<iframe>`. -- `iframe.contentDocument` is a reference to the document inside the `<iframe>`. - -When we access an embedded window, the browser checks if the iframe has the same origin. If that's not so then the access is denied (with exclusions noted above). - -For instance, here's an `<iframe>` from another origin: +For instance, let's try reading and writing to `<iframe>` from another origin: ```html run <iframe src="https://example.com" id="iframe"></iframe> @@ -64,44 +45,47 @@ For instance, here's an `<iframe>` from another origin: <script> iframe.onload = function() { // we can get the reference to the inner window - let iframeWindow = iframe.contentWindow; - +*!* + let iframeWindow = iframe.contentWindow; // OK +*/!* try { // ...but not to the document inside it - let doc = iframe.contentDocument; +*!* + let doc = iframe.contentDocument; // ERROR +*/!* } catch(e) { alert(e); // Security Error (another origin) } - // also we can't read the URL of the page in it + // also we can't READ the URL of the page in iframe try { - alert(iframe.contentWindow.location); + // Can't read URL from the Location object +*!* + let href = iframe.contentWindow.location.href; // ERROR +*/!* } catch(e) { alert(e); // Security Error } - // ...but we can change it (and thus load something else into the iframe)! - iframe.contentWindow.location = '/'; // works + // ...we can WRITE into location (and thus load something else into the iframe)! +*!* + iframe.contentWindow.location = '/'; // OK +*/!* - iframe.onload = null; // clear the handler, to run this code only once + iframe.onload = null; // clear the handler, not to run it after the location change }; </script> ``` The code above shows errors for any operations except: -- Getting the reference to the inner window `iframe.contentWindow` -- Changing its `location`. - -```smart header="`iframe.onload` vs `iframe.contentWindow.onload`" -The `iframe.onload` event is actually the same as `iframe.contentWindow.onload`. It triggers when the embedded window fully loads with all resources. - -...But `iframe.onload` is always available, while `iframe.contentWindow.onload` needs the same origin. -``` +- Getting the reference to the inner window `iframe.contentWindow` - that's allowed. +- Writing to `location`. -And now an example with the same origin. We can do anything with the embedded window: +Contrary to that, if the `<iframe>` has the same origin, we can do anything with it: ```html run +<!-- iframe from the same site --> <iframe src="/" id="iframe"></iframe> <script> @@ -112,9 +96,40 @@ And now an example with the same origin. We can do anything with the embedded wi </script> ``` -### Please wait until the iframe loads +```smart header="`iframe.onload` vs `iframe.contentWindow.onload`" +The `iframe.onload` event (on the `<iframe>` tag) is essentially the same as `iframe.contentWindow.onload` (on the embedded window object). It triggers when the embedded window fully loads with all resources. + +...But we can't access `iframe.contentWindow.onload` for an iframe from another origin, so using `iframe.onload`. +``` + +## Windows on subdomains: document.domain + +By definition, two URLs with different domains have different origins. + +But if windows share the same second-level domain, for instance `john.site.com`, `peter.site.com` and `site.com` (so that their common second-level domain is `site.com`), we can make the browser ignore that difference, so that they can be treated as coming from the "same origin" for the purposes of cross-window communication. + +To make it work, each such window should run the code: + +```js +document.domain = 'site.com'; +``` + +That's all. Now they can interact without limitations. Again, that's only possible for pages with the same second-level domain. + +```warn header="Deprecated, but still working" +The `document.domain` property is in the process of being removed from the [specification](https://html.spec.whatwg.org/multipage/origin.html#relaxing-the-same-origin-restriction). The cross-window messaging (explained soon below) is the suggested replacement. + +That said, as of now all browsers support it. And the support will be kept for the future, not to break old code that relies on `document.domain`. +``` + + +## Iframe: wrong document pitfall -When an iframe is created, it immediately has a document. But that document is different from the one that finally loads into it! +When an iframe comes from the same origin, and we may access its `document`, there's a pitfall. It's not related to cross-origin things, but important to know. + +Upon its creation an iframe immediately has a document. But that document is different from the one that loads into it! + +So if we do something with the document immediately, that will probably be lost. Here, look: @@ -134,11 +149,13 @@ Here, look: </script> ``` -That's actually a well-known pitfall for developers. We shouldn't work with the document immediately, because that's the *wrong document*. If we set any event handlers on it, they will be ignored. +We shouldn't work with the document of a not-yet-loaded iframe, because that's the *wrong document*. If we set any event handlers on it, they will be ignored. + +How to detect the moment when the document is there? -...But the `onload` event triggers when the whole iframe with all resources is loaded. What if we want to act sooner, on `DOMContentLoaded` of the embedded document? +The right document is definitely at place when `iframe.onload` triggers. But it only triggers when the whole iframe with all resources is loaded. -That's not possible if the iframe comes from another origin. But for the same origin we can try to catch the moment when a new document appears, and then setup necessary handlers, like this: +We can try to catch the moment earlier using checks in `setInterval`: ```html run <iframe src="/" id="iframe"></iframe> @@ -148,21 +165,17 @@ That's not possible if the iframe comes from another origin. But for the same or // every 100 ms check if the document is the new one let timer = setInterval(() => { - if (iframe.contentDocument == oldDoc) return; + let newDoc = iframe.contentDocument; + if (newDoc == oldDoc) return; - // new document, let's set handlers - iframe.contentDocument.addEventListener('DOMContentLoaded', () => { - iframe.contentDocument.body.prepend('Hello, world!'); - }); + alert("New document is here!"); clearInterval(timer); // cancel setInterval, don't need it any more }, 100); </script> ``` -Let me know in comments if you know a better solution here. - -## window.frames +## Collection: window.frames An alternative way to get a window object for `<iframe>` -- is to get it from the named collection `window.frames`: @@ -204,11 +217,11 @@ if (window == top) { // current window == window.top? } ``` -## The sandbox attribute +## The "sandbox" iframe attribute The `sandbox` attribute allows for the exclusion of certain actions inside an `<iframe>` in order to prevent it executing untrusted code. It "sandboxes" the iframe by treating it as coming from another origin and/or applying other limitations. -By default, for `<iframe sandbox src="...">` the "default set" of restrictions is applied to the iframe. But we can provide a space-separated list of "excluded" limitations as a value of the attribute, like this: `<iframe sandbox="allow-forms allow-popups">`. The listed limitations are not applied. +There's a "default set" of restrictions applied for `<iframe sandbox src="...">`. But it can be relaxed if we provide a space-separated list of restrictions that should not be applied as a value of the attribute, like this: `<iframe sandbox="allow-forms allow-popups">`. In other words, an empty `"sandbox"` attribute puts the strictest limitations possible, but we can put a space-delimited list of those that we want to lift. @@ -257,14 +270,14 @@ The window that wants to send a message calls [postMessage](mdn:api/Window.postM Arguments: `data` -: The data to send. Can be any object, the data is cloned using the "structured cloning algorithm". IE supports only strings, so we should `JSON.stringify` complex objects to support that browser. +: The data to send. Can be any object, the data is cloned using the "structured serialization algorithm". IE supports only strings, so we should `JSON.stringify` complex objects to support that browser. `targetOrigin` : Specifies the origin for the target window, so that only a window from the given origin will get the message. -The `targetOrigin` is a safety measure. Remember, if the target window comes from another origin, we can't read it's `location`. So we can't be sure which site is open in the intended window right now: the user could navigate away. +The `targetOrigin` is a safety measure. Remember, if the target window comes from another origin, we can't read its `location` in the sender window. So we can't be sure which site is open in the intended window right now: the user could navigate away, and the sender window has no idea about it. -Specifying `targetOrigin` ensures that the window only receives the data if it's still at that site. Good when the data is sensitive. +Specifying `targetOrigin` ensures that the window only receives the data if it's still at the right site. Important when the data is sensitive. For instance, here `win` will only receive the message if it has a document from the origin `http://example.com`: @@ -306,7 +319,7 @@ The event object has special properties: : The origin of the sender, for instance `http://javascript.info`. `source` -: The reference to the sender window. We can immediately `postMessage` back if we want. +: The reference to the sender window. We can immediately `source.postMessage(...)` back if we want. To assign that handler, we should use `addEventListener`, a short syntax `window.onmessage` does not work. @@ -320,6 +333,8 @@ window.addEventListener("message", function(event) { } alert( "received: " + event.data ); + + // can message back using event.source.postMessage(...) }); ``` @@ -327,17 +342,13 @@ The full example: [codetabs src="postmessage" height=120] -```smart header="There's no delay" -There's totally no delay between `postMessage` and the `message` event. That happens synchronously, even faster than `setTimeout(...,0)`. -``` - ## Summary To call methods and access the content of another window, we should first have a reference to it. -For popups we have two properties: -- `window.open` -- opens a new window and returns a reference to it, -- `window.opener` -- a reference to the opener window from a popup +For popups we have these references: +- From the opener window: `window.open` -- opens a new window and returns a reference to it, +- From the popup: `window.opener` -- is a reference to the opener window from a popup. For iframes, we can access parent/children windows using: - `window.frames` -- a collection of nested window objects, @@ -347,18 +358,17 @@ For iframes, we can access parent/children windows using: If windows share the same origin (host, port, protocol), then windows can do whatever they want with each other. Otherwise, only possible actions are: -- Change the location of another window (write-only access). +- Change the `location` of another window (write-only access). - Post a message to it. - -Exclusions are: +Exceptions are: - Windows that share the same second-level domain: `a.site.com` and `b.site.com`. Then setting `document.domain='site.com'` in both of them puts them into the "same origin" state. - If an iframe has a `sandbox` attribute, it is forcefully put into the "different origin" state, unless the `allow-same-origin` is specified in the attribute value. That can be used to run untrusted code in iframes from the same site. -The `postMessage` interface allows two windows to talk with security checks: +The `postMessage` interface allows two windows with any origins to talk: 1. The sender calls `targetWin.postMessage(data, targetOrigin)`. -2. If `targetOrigin` is not `'*'`, then the browser checks if window `targetWin` has the URL from `targetWin` site. +2. If `targetOrigin` is not `'*'`, then the browser checks if window `targetWin` has the origin `targetOrigin`. 3. If it is so, then `targetWin` triggers the `message` event with special properties: - `origin` -- the origin of the sender window (like `http://my.site.com`) - `source` -- the reference to the sender window. diff --git a/3-frames-and-windows/03-cross-window-communication/sandbox.view/index.html b/3-frames-and-windows/03-cross-window-communication/sandbox.view/index.html index 478830610..46dd7b5cc 100644 --- a/3-frames-and-windows/03-cross-window-communication/sandbox.view/index.html +++ b/3-frames-and-windows/03-cross-window-communication/sandbox.view/index.html @@ -7,7 +7,7 @@ <body> - <div>The iframe below is has <code>sandbox</code> attribute.</div> + <div>The iframe below has the <code>sandbox</code> attribute.</div> <iframe sandbox src="sandboxed.html" style="height:60px;width:90%"></iframe> diff --git a/3-frames-and-windows/06-clickjacking/article.md b/3-frames-and-windows/06-clickjacking/article.md index 9752835e5..34d0a91ae 100644 --- a/3-frames-and-windows/06-clickjacking/article.md +++ b/3-frames-and-windows/06-clickjacking/article.md @@ -57,12 +57,12 @@ Here's the same example, but closer to reality, with `opacity:0` for `<iframe>`: [codetabs src="clickjacking" height=160] -All we need to attack -- is to position the `<iframe>` on the evil page in such a way that the button is right over the link. That's usually possible with CSS. +All we need to attack -- is to position the `<iframe>` on the evil page in such a way that the button is right over the link. So that when a user clicks the link, they actually click the button. That's usually doable with CSS. ```smart header="Clickjacking is for clicks, not for keyboard" -The attack only affects mouse actions. +The attack only affects mouse actions (or similar, like taps on mobile). -Technically, if we have a text field to hack, then we can position an iframe in such a way that text fields overlap each other. So when a visitor tries to focus on the input they see on the page, they actually focus on the input inside the iframe. +Keyboard input is much difficult to redirect. Technically, if we have a text field to hack, then we can position an iframe in such a way that text fields overlap each other. So when a visitor tries to focus on the input they see on the page, they actually focus on the input inside the iframe. But then there's a problem. Everything that the visitor types will be hidden, because the iframe is not visible. @@ -87,19 +87,19 @@ This not a reliable defence, because there are many ways to hack around it. Let' ### Blocking top-navigation -We can block the transition caused by changing `top.location` in the [beforeunload](info:onload-ondomcontentloaded#window.onbeforeunload) event. +We can block the transition caused by changing `top.location` in [beforeunload](info:onload-ondomcontentloaded#window.onbeforeunload) event handler. -The top page (belonging to the hacker) sets a handler to it, and when the `iframe` tries to change `top.location` the visitor gets a message asking them whether they want to leave. +The top page (enclosing one, belonging to the hacker) sets a preventing handler to it, like this: -Like this: ```js window.onbeforeunload = function() { - window.onbeforeunload = null; - return "Want to leave without learning all the secrets (he-he)?"; + return false; }; ``` -In most cases the visitor would answer negatively because they don't know about the iframe - all they can see is the top page, leading them to think there is no reason to leave. So `top.location` won't change! +When the `iframe` tries to change `top.location`, the visitor gets a message asking them whether they want to leave. + +In most cases the visitor would answer negatively because they don't know about the iframe - all they can see is the top page, there's no reason to leave. So `top.location` won't change! In action: @@ -123,7 +123,7 @@ There are other ways to work around that simple protection too. The server-side header `X-Frame-Options` can permit or forbid displaying the page inside a frame. -It must be sent *by the server*: the browser will ignore it if found in a `<meta>` tag. So, `<meta http-equiv="X-Frame-Options"...>` won't do anything. +It must be sent exactly as HTTP-header: the browser will ignore it if found in HTML `<meta>` tag. So, `<meta http-equiv="X-Frame-Options"...>` won't do anything. The header may have 3 values: @@ -154,9 +154,9 @@ Depending on your browser, the `iframe` above is either empty or alerting you th ## Showing with disabled functionality -The `X-Frame-Options` header has a side-effect. Other sites won't be able to show our page in a frame, even if they have good reasons to do so. +The `X-Frame-Options` header has a side effect. Other sites won't be able to show our page in a frame, even if they have good reasons to do so. -So there are other solutions... For instance, we can "cover" the page with a `<div>` with `height: 100%; width: 100%;`, so that it intercepts all clicks. That `<div>` should disappear if `window == top` or if we figure out that we don't need the protection. +So there are other solutions... For instance, we can "cover" the page with a `<div>` with styles `height: 100%; width: 100%;`, so that it will intercept all clicks. That `<div>` is to be removed if `window == top` or if we figure out that we don't need the protection. Something like this: @@ -191,43 +191,25 @@ The demo: ## Samesite cookie attribute -The `samesite` cookie attribute can also prevent clickjacking attacks. The purpose of the attribute is to prevent cookies from being sent to a website when the user doesn't intend to visit the website. It is designed to prevent cross-site request forgery attacks, but also helps with clickjacking because a hijacked click usually results in an unintended request to a different site. When a cookie has the `samesite` attribute, whether the value is `strict` or `lax`, cookies are not sent to a website when it is loaded inside an iframe. - -The `samesite` attribute can be set using HTTP response headers or JavaScript. Via HTTP, it looks like: - -`Set-Cookie: demoCookie=demoValue; samesite=lax` - -or +The `samesite` cookie attribute can also prevent clickjacking attacks. -`Set-Cookie: demoCookie=demoValue; samesite=strict` +A cookie with such attribute is only sent to a website if it's opened directly, not via a frame, or otherwise. More information in the chapter <info:cookie#samesite>. -In JavaScript, it is: +If the site, such as Facebook, had `samesite` attribute on its authentication cookie, like this: -```html -document.cookie = "demoCookie=demoValue; SameSite=Lax"; -document.cookie = "demoCookie=demoValue; SameSite=Strict"; +``` +Set-Cookie: authorization=secret; samesite ``` -When the value is `lax`, these types of requests are blocked: -- Form POST submit (<form method="POST" action="...">) -- iframe (<iframe src="..."></iframe>) -- AJAX ($.get("...")) -- Image (<img src="...">) -- Script (<script src="..."></script>) -- Stylesheet (<link rel="stylesheet" type="text/css" href="...">) - -When the value is `strict`, these types of requests are also blocked, in addition to those under `lax`: -- Clicking a link (<a href="..."></a>) -- Prerender (<link rel="prerender" href=".."/>) -- Form GET submit (<form method="GET" action="...">) +...Then such cookie wouldn't be sent when Facebook is open in iframe from another site. So the attack would fail. -In this case, we are concerned with iframe requests. A clickjacking attempt would fail because the user is not considered logged into, for example, Facebook, so they can't "Like" anything through the iframe. +The `samesite` cookie attribute will not have an effect when cookies are not used. This may allow other websites to easily show our public, unauthenticated pages in iframes. -The `samesite` attribute will not have an effect when cookies are not used. This may allow websites to easily show public, unauthenticated pages in iframes on unaffiliated websites. However, this may also allow clickjacking attacks to work in a few limited cases. An anonymous polling website that prevents duplicate voting by checking IP addresses, for example, would still be vulnerable to clickjacking because it does not authenticate users using cookies. +However, this may also allow clickjacking attacks to work in a few limited cases. An anonymous polling website that prevents duplicate voting by checking IP addresses, for example, would still be vulnerable to clickjacking because it does not authenticate users using cookies. ## Summary -Clickjacking is a way to "trick" users into clicking on a malicious site without even knowing what's happening. That's dangerous if there are important click-activated actions. +Clickjacking is a way to "trick" users into clicking on a victim site without even knowing what's happening. That's dangerous if there are important click-activated actions. A hacker can post a link to their evil page in a message, or lure visitors to their page by some other means. There are many variations. diff --git a/4-binary/01-arraybuffer-binary-arrays/01-concat/_js.view/solution.js b/4-binary/01-arraybuffer-binary-arrays/01-concat/_js.view/solution.js index 2f51384ef..00c37bb94 100644 --- a/4-binary/01-arraybuffer-binary-arrays/01-concat/_js.view/solution.js +++ b/4-binary/01-arraybuffer-binary-arrays/01-concat/_js.view/solution.js @@ -2,9 +2,9 @@ function concat(arrays) { // sum of individual array lengths let totalLength = arrays.reduce((acc, value) => acc + value.length, 0); - if (!arrays.length) return null; - let result = new Uint8Array(totalLength); + + if (!arrays.length) return result; // for each array - copy it over result // next array is copied right after the previous one diff --git a/4-binary/01-arraybuffer-binary-arrays/article.md b/4-binary/01-arraybuffer-binary-arrays/article.md index 5a0482bbb..2827e277e 100644 --- a/4-binary/01-arraybuffer-binary-arrays/article.md +++ b/4-binary/01-arraybuffer-binary-arrays/article.md @@ -9,7 +9,7 @@ Although, there's a bit of confusion, because there are many classes. To name a Binary data in JavaScript is implemented in a non-standard way, compared to other languages. But when we sort things out, everything becomes fairly simple. -**The basic binary object is `ArrayBuffer` -- a reference to a fixed-length contiguos memory area.** +**The basic binary object is `ArrayBuffer` -- a reference to a fixed-length contiguous memory area.** We create it like this: ```js run @@ -30,11 +30,11 @@ Let's eliminate a possible source of confusion. `ArrayBuffer` has nothing in com **To manipulate an `ArrayBuffer`, we need to use a "view" object.** -A view object does not store anything on it's own. It's the "eyeglasses" that give an interpretation of the bytes stored in the `ArrayBuffer`. +A view object does not store anything on its own. It's the "eyeglasses" that give an interpretation of the bytes stored in the `ArrayBuffer`. For instance: -- **`Uint8Array`** -- treats each byte in `ArrayBuffer` as a separate number, with possible values are from 0 to 255 (a byte is 8-bit, so it can hold only that much). Such value is called a "8-bit unsigned integer". +- **`Uint8Array`** -- treats each byte in `ArrayBuffer` as a separate number, with possible values from 0 to 255 (a byte is 8-bit, so it can hold only that much). Such value is called a "8-bit unsigned integer". - **`Uint16Array`** -- treats every 2 bytes as an integer, with possible values from 0 to 65535. That's called a "16-bit unsigned integer". - **`Uint32Array`** -- treats every 4 bytes as an integer, with possible values from 0 to 4294967295. That's called a "32-bit unsigned integer". - **`Float64Array`** -- treats every 8 bytes as a floating point number with possible values from <code>5.0x10<sup>-324</sup></code> to <code>1.8x10<sup>308</sup></code>. @@ -71,10 +71,13 @@ for(let num of view) { ## TypedArray -The common term for all these views (`Uint8Array`, `Uint32Array`, etc) is [TypedArray](https://tc39.github.io/ecma262/#sec-typedarray-objects). They share the same set of methods and properities. +The common term for all these views (`Uint8Array`, `Uint32Array`, etc) is [TypedArray](https://tc39.github.io/ecma262/#sec-typedarray-objects). They share the same set of methods and properties. -They are much more like regular arrays: have indexes and iterable. +Please note, there's no constructor called `TypedArray`, it's just a common "umbrella" term to represent one of views over `ArrayBuffer`: `Int8Array`, `Uint8Array` and so on, the full list will soon follow. +When you see something like `new TypedArray`, it means any of `new Int8Array`, `new Uint8Array`, etc. + +Typed arrays behave like regular arrays: have indexes and are iterable. A typed array constructor (be it `Int8Array` or `Float64Array`, doesn't matter) behaves differently depending on argument types. @@ -99,17 +102,17 @@ new TypedArray(); *!* let arr = new Uint8Array([0, 1, 2, 3]); */!* - alert( arr.length ); // 4 - alert( arr[1] ); // 1 + alert( arr.length ); // 4, created binary array of the same length + alert( arr[1] ); // 1, filled with 4 bytes (unsigned 8-bit integers) with given values ``` -3. If another `TypedArray` is supplied, it does the same: creates a typed array of the same length and copies values. Values are converted to the new type in the process. +3. If another `TypedArray` is supplied, it does the same: creates a typed array of the same length and copies values. Values are converted to the new type in the process, if needed. ```js run let arr16 = new Uint16Array([1, 1000]); *!* let arr8 = new Uint8Array(arr16); */!* alert( arr8[0] ); // 1 - alert( arr8[1] ); // 232 (tried to copy 1000, but can't fit 1000 into 8 bits) + alert( arr8[1] ); // 232, tried to copy 1000, but can't fit 1000 into 8 bits (explanations below) ``` 4. For a numeric argument `length` -- creates the typed array to contain that many elements. Its byte length will be `length` multiplied by the number of bytes in a single item `TypedArray.BYTES_PER_ELEMENT`: @@ -123,9 +126,9 @@ new TypedArray(); We can create a `TypedArray` directly, without mentioning `ArrayBuffer`. But a view cannot exist without an underlying `ArrayBuffer`, so gets created automatically in all these cases except the first one (when provided). -To access the `ArrayBuffer`, there are properties: -- `arr.buffer` -- references the `ArrayBuffer`. -- `arr.byteLength` -- the length of the `ArrayBuffer`. +To access the underlying `ArrayBuffer`, there are following properties in `TypedArray`: +- `buffer` -- references the `ArrayBuffer`. +- `byteLength` -- the length of the `ArrayBuffer`. So, we can always move from one view to another: ```js @@ -206,10 +209,10 @@ These methods allow us to copy typed arrays, mix them, create new arrays from ex ## DataView -[DataView](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DataView) is a special super-flexible "untyped" view over `ArrayBuffer`. It allows to access the data on any offset in any format. +[DataView](mdn:/JavaScript/Reference/Global_Objects/DataView) is a special super-flexible "untyped" view over `ArrayBuffer`. It allows to access the data on any offset in any format. - For typed arrays, the constructor dictates what the format is. The whole array is supposed to be uniform. The i-th number is `arr[i]`. -- With `DataView` we access the data with methods like `.getUint8(i)` or `.gteUint16(i)`. We choose the format at method call time instead of the construction time. +- With `DataView` we access the data with methods like `.getUint8(i)` or `.getUint16(i)`. We choose the format at method call time instead of the construction time. The syntax: @@ -224,6 +227,7 @@ new DataView(buffer, [byteOffset], [byteLength]) For instance, here we extract numbers in different formats from the same buffer: ```js run +// binary array of 4 bytes, all have the maximal value 255 let buffer = new Uint8Array([255, 255, 255, 255]).buffer; let dataView = new DataView(buffer); @@ -231,16 +235,16 @@ let dataView = new DataView(buffer); // get 8-bit number at offset 0 alert( dataView.getUint8(0) ); // 255 -// now get 16-bit number at offset 0, that's 2 bytes, both with max value +// now get 16-bit number at offset 0, it consists of 2 bytes, together interpreted as 65535 alert( dataView.getUint16(0) ); // 65535 (biggest 16-bit unsigned int) // get 32-bit number at offset 0 alert( dataView.getUint32(0) ); // 4294967295 (biggest 32-bit unsigned int) -dataView.setUint32(0, 0); // set 4-byte number to zero +dataView.setUint32(0, 0); // set 4-byte number to zero, thus setting all bytes to 0 ``` -`DataView` is great when we store mixed-format data in the same buffer. E.g we store a sequence of pairs (16-bit integer, 32-bit float). Then `DataView` allows to access them easily. +`DataView` is great when we store mixed-format data in the same buffer. For example, when we store a sequence of pairs (16-bit integer, 32-bit float), `DataView` allows to access them easily. ## Summary @@ -249,20 +253,19 @@ dataView.setUint32(0, 0); // set 4-byte number to zero To do almost any operation on `ArrayBuffer`, we need a view. - It can be a `TypedArray`: - - `Uint8Array`, `Uint16Array`, `Uint32Array` -- for integer numbers of 8, 16 and 32 bits. + - `Uint8Array`, `Uint16Array`, `Uint32Array` -- for unsigned integers of 8, 16, and 32 bits. - `Uint8ClampedArray` -- for 8-bit integers, "clamps" them on assignment. - `Int8Array`, `Int16Array`, `Int32Array` -- for signed integer numbers (can be negative). - `Float32Array`, `Float64Array` -- for signed floating-point numbers of 32 and 64 bits. - Or a `DataView` -- the view that uses methods to specify a format, e.g. `getUint8(offset)`. -In most cases we create and operate directly on typed arrays, leaving `ArrayBuffer` under cover, as a "common discriminator". We can access it as `.buffer` and make another view if needed. +In most cases we create and operate directly on typed arrays, leaving `ArrayBuffer` under cover, as a "common denominator". We can access it as `.buffer` and make another view if needed. -There are also two additional terms: +There are also two additional terms, that are used in descriptions of methods that operate on binary data: - `ArrayBufferView` is an umbrella term for all these kinds of views. - `BufferSource` is an umbrella term for `ArrayBuffer` or `ArrayBufferView`. -These are used in descriptions of methods that operate on binary data. `BufferSource` is one of the most common teerms, as it means "any kind of binary data" -- an `ArrayBuffer` or a view over it. - +We'll see these terms in the next chapters. `BufferSource` is one of the most common terms, as it means "any kind of binary data" -- an `ArrayBuffer` or a view over it. Here's a cheatsheet: diff --git a/4-binary/02-text-decoder/article.md b/4-binary/02-text-decoder/article.md index 9287279ec..a0c80145c 100644 --- a/4-binary/02-text-decoder/article.md +++ b/4-binary/02-text-decoder/article.md @@ -2,7 +2,7 @@ What if the binary data is actually a string? For instance, we received a file with textual data. -The build-in [TextDecoder](https://encoding.spec.whatwg.org/#interface-textdecoder) object allows to read the value into an an actual JavaScript string, given the buffer and the encoding. +The built-in [TextDecoder](https://encoding.spec.whatwg.org/#interface-textdecoder) object allows one to read the value into an actual JavaScript string, given the buffer and the encoding. We first need to create it: ```js @@ -12,7 +12,7 @@ let decoder = new TextDecoder([label], [options]); - **`label`** -- the encoding, `utf-8` by default, but `big5`, `windows-1251` and many other are also supported. - **`options`** -- optional object: - **`fatal`** -- boolean, if `true` then throw an exception for invalid (non-decodable) characters, otherwise (default) replace them with character `\uFFFD`. - - **`ignoreBOM`** -- boolean, if `true` then ignore BOM (an optional byte-order unicode mark), rarely needed. + - **`ignoreBOM`** -- boolean, if `true` then ignore BOM (an optional byte-order Unicode mark), rarely needed. ...And then decode: @@ -58,7 +58,7 @@ alert( new TextDecoder().decode(binaryString) ); // Hello The syntax is: -```js run +```js let encoder = new TextEncoder(); ``` diff --git a/4-binary/03-blob/article.md b/4-binary/03-blob/article.md index 0bf1ffe78..a6ef50c33 100644 --- a/4-binary/03-blob/article.md +++ b/4-binary/03-blob/article.md @@ -2,13 +2,16 @@ `ArrayBuffer` and views are a part of ECMA standard, a part of JavaScript. -In the browser, there are additional higher-level objects, described in [File API](https://www.w3.org/TR/FileAPI/). +In the browser, there are additional higher-level objects, described in [File API](https://www.w3.org/TR/FileAPI/), in particular `Blob`. -`Blob` consists of an optional string `type` (a MIME-type usually), plus `blobParts` -- a sequence of other `Blob` objects, strings and `BufferSources`. +`Blob` consists of an optional string `type` (a MIME-type usually), plus `blobParts` -- a sequence of other `Blob` objects, strings and `BufferSource`.  +<<<<<<< HEAD Thanks to `type`, we can download/upload blobs, and it naturally becomes `Content-Type` in network requests. +======= +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b The constructor syntax is: @@ -18,8 +21,8 @@ new Blob(blobParts, options); - **`blobParts`** is an array of `Blob`/`BufferSource`/`String` values. - **`options`** optional object: - - **`type`** -- blob type, usually MIME-type, e.g. `image/png`, - - **`endings`** -- whether to transform end-of-line to make the blob correspond to current OS newlines (`\r\n` or `\n`). By default `"transparent"` (do nothing), but also can be `"native"` (transform). + - **`type`** -- `Blob` type, usually MIME-type, e.g. `image/png`, + - **`endings`** -- whether to transform end-of-line to make the `Blob` correspond to current OS newlines (`\r\n` or `\n`). By default `"transparent"` (do nothing), but also can be `"native"` (transform). For example: @@ -31,13 +34,13 @@ let blob = new Blob(["<html>…</html>"], {type: 'text/html'}); ```js // create Blob from a typed array and strings -let hello = new Uint8Array([72, 101, 108, 108, 111]); // "hello" in binary form +let hello = new Uint8Array([72, 101, 108, 108, 111]); // "Hello" in binary form let blob = new Blob([hello, ' ', 'world'], {type: 'text/plain'}); ``` -We can extract blob slices with: +We can extract `Blob` slices with: ```js blob.slice([byteStart], [byteEnd], [contentType]); @@ -49,17 +52,19 @@ blob.slice([byteStart], [byteEnd], [contentType]); The arguments are similar to `array.slice`, negative numbers are allowed too. -```smart header="Blobs are immutable" -We can't change data directly in a blob, but we can slice parts of blobs, create new blobs from them, mix them into a new blob and so on. +```smart header="`Blob` objects are immutable" +We can't change data directly in a `Blob`, but we can slice parts of a `Blob`, create new `Blob` objects from them, mix them into a new `Blob` and so on. This behavior is similar to JavaScript strings: we can't change a character in a string, but we can make a new corrected string. ``` ## Blob as URL -A Blob can be easily used as an URL for `<a>`, `<img>` or other tags, to show its contents. +A Blob can be easily used as a URL for `<a>`, `<img>` or other tags, to show its contents. -Let's start with a simple example. By clicking on a link you download a dynamically-generated blob with `hello world` contents as a file: +Thanks to `type`, we can also download/upload `Blob` objects, and the `type` naturally becomes `Content-Type` in network requests. + +Let's start with a simple example. By clicking on a link you download a dynamically-generated `Blob` with `hello world` contents as a file: ```html run <!-- download attribute forces the browser to download instead of navigating --> @@ -74,7 +79,7 @@ link.href = URL.createObjectURL(blob); We can also create a link dynamically in JavaScript and simulate a click by `link.click()`, then download starts automatically. -Here's the similar "on the fly" blob creation and download code, but without HTML: +Here's the similar code that causes user to download the dynamically created `Blob`, without any HTML: ```js run let link = document.createElement('a'); @@ -89,35 +94,37 @@ link.click(); URL.revokeObjectURL(link.href); ``` -**`URL.createObjectURL` takes a blob and creates an unique URL for it, in the form `blob:<origin>/<uuid>`.** +`URL.createObjectURL` takes a `Blob` and creates a unique URL for it, in the form `blob:<origin>/<uuid>`. -That's what the generated url looks like: +That's what the value of `link.href` looks like: ``` blob:https://javascript.info/1e67e00e-860d-40a5-89ae-6ab0cbee6273 ``` -The browser for each url generated by `URL.createObjectURL` stores an the url -> blob mapping internally. So such urls are short, but allow to access the blob. +For each URL generated by `URL.createObjectURL` the browser stores a URL -> `Blob` mapping internally. So such URLs are short, but allow to access the `Blob`. + +A generated URL (and hence the link with it) is only valid within the current document, while it's open. And it allows to reference the `Blob` in `<img>`, `<a>`, basically any other object that expects a URL. -A generated url is only valid while the current document is open. And it allows to reference the blob in `<img>`, `<a>`, any other object that expects an url. +There's a side effect though. While there's a mapping for a `Blob`, the `Blob` itself resides in the memory. The browser can't free it. -There's a side-effect though. While there's an mapping for a blob, the blob itself resides in the memory. The browser can't free it. +The mapping is automatically cleared on document unload, so `Blob` objects are freed then. But if an app is long-living, then that doesn't happen soon. -The mapping is automatically cleared on document unload, so blobs are freed then. But if an app is long-living, then that doesn't happen soon. So if we create an URL, that blob will hang in memory, even if not needed any more. +**So if we create a URL, that `Blob` will hang in memory, even if not needed any more.** -**`URL.revokeObjectURL(url)` removes the reference from the internal mapping, thus allowing the blob to be deleted (if there are no other references), and the memory to be freed.** +`URL.revokeObjectURL(url)` removes the reference from the internal mapping, thus allowing the `Blob` to be deleted (if there are no other references), and the memory to be freed. -In the last example, we intend the blob to be used only once, for instant downloading, so we call `URL.revokeObjectURL(link.href)` immediately. +In the last example, we intend the `Blob` to be used only once, for instant downloading, so we call `URL.revokeObjectURL(link.href)` immediately. -In the previous example though, with the clickable HTML-link, we don't call `URL.revokeObjectURL(link.href)`, because that would make the blob url invalid. After the revocation, as the mapping is removed, the url doesn't work any more. +In the previous example with the clickable HTML-link, we don't call `URL.revokeObjectURL(link.href)`, because that would make the `Blob` url invalid. After the revocation, as the mapping is removed, the URL doesn't work any more. ## Blob to base64 -An alternative to `URL.createObjectURL` is to convert a blob into a base64-encoded string. +An alternative to `URL.createObjectURL` is to convert a `Blob` into a base64-encoded string. That encoding represents binary data as a string of ultra-safe "readable" characters with ASCII-codes from 0 to 64. And what's more important -- we can use this encoding in "data-urls". -A [data url](https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs) has the form `data:[<mediatype>][;base64],<data>`. We can use such urls everywhere, on par with "regular" urls. +A [data url](mdn:/http/Data_URIs) has the form `data:[<mediatype>][;base64],<data>`. We can use such urls everywhere, on par with "regular" urls. For instance, here's a smiley: @@ -128,7 +135,7 @@ For instance, here's a smiley: The browser will decode the string and show the image: <img src=""> -To transform a blob into base64, we'll use the built-in `FileReader` object. It can read data from Blobs in multiple formats. In the [next chapter](info:file) we'll cover it more in-depth. +To transform a `Blob` into base64, we'll use the built-in `FileReader` object. It can read data from Blobs in multiple formats. In the [next chapter](info:file) we'll cover it more in-depth. Here's the demo of downloading a blob, now via base-64: @@ -149,23 +156,23 @@ reader.onload = function() { }; ``` -Both ways of making an URL of a blob are usable. But usually `URL.createObjectURL(blob)` is simpler and faster. +Both ways of making a URL of a `Blob` are usable. But usually `URL.createObjectURL(blob)` is simpler and faster. ```compare title-plus="URL.createObjectURL(blob)" title-minus="Blob to data url" + We need to revoke them if care about memory. + Direct access to blob, no "encoding/decoding" - No need to revoke anything. -- Performance and memory losses on big blobs for encoding. +- Performance and memory losses on big `Blob` objects for encoding. ``` ## Image to blob -We can create a blob of an image, an image part, or even make a page screenshot. That's handy to upload it somewhere. +We can create a `Blob` of an image, an image part, or even make a page screenshot. That's handy to upload it somewhere. Image operations are done via `<canvas>` element: -1. Draw an image (or its part) on canvas using [canvas.drawImage](https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/drawImage). -2. Call canvas method [.toBlob(callback, format, quality)](https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob) that creates a blob and runs `callback` with it when done. +1. Draw an image (or its part) on canvas using [canvas.drawImage](mdn:/api/CanvasRenderingContext2D/drawImage). +2. Call canvas method [.toBlob(callback, format, quality)](mdn:/api/HTMLCanvasElement/toBlob) that creates a `Blob` and runs `callback` with it when done. In the example below, an image is just copied, but we could cut from it, or transform it on canvas prior to making a blob: @@ -184,7 +191,7 @@ let context = canvas.getContext('2d'); context.drawImage(img, 0, 0); // we can context.rotate(), and do many other things on canvas -// toBlob is async opereation, callback is called when done +// toBlob is async operation, callback is called when done canvas.toBlob(function(blob) { // blob ready, download it let link = document.createElement('a'); @@ -203,27 +210,50 @@ If we prefer `async/await` instead of callbacks: let blob = await new Promise(resolve => canvasElem.toBlob(resolve, 'image/png')); ``` -For screenshotting a page, we can use a library such as <https://github.com/niklasvh/html2canvas>. What it does is just walks the page and draws it on `<canvas>`. Then we can get a blob of it the same way as above. +For screenshotting a page, we can use a library such as <https://github.com/niklasvh/html2canvas>. What it does is just walks the page and draws it on `<canvas>`. Then we can get a `Blob` of it the same way as above. ## From Blob to ArrayBuffer The `Blob` constructor allows to create a blob from almost anything, including any `BufferSource`. -But if we need to perform low-level processing, we can get the lowest-level `ArrayBuffer` from it using `FileReader`: +But if we need to perform low-level processing, we can get the lowest-level `ArrayBuffer` from `blob.arrayBuffer()`: ```js // get arrayBuffer from blob -let fileReader = new FileReader(); +const bufferPromise = await blob.arrayBuffer(); -*!* -fileReader.readAsArrayBuffer(blob); -*/!* - -fileReader.onload = function(event) { - let arrayBuffer = fileReader.result; -}; +// or +blob.arrayBuffer().then(buffer => /* process the ArrayBuffer */); ``` +## From Blob to stream + +When we read and write to a blob of more than `2 GB`, the use of `arrayBuffer` becomes more memory intensive for us. At this point, we can directly convert the blob to a stream. + +A stream is a special object that allows to read from it (or write into it) portion by portion. It's outside of our scope here, but here's an example, and you can read more at <https://developer.mozilla.org/en-US/docs/Web/API/Streams_API>. Streams are convenient for data that is suitable for processing piece-by-piece. + +The `Blob` interface's `stream()` method returns a `ReadableStream` which upon reading returns the data contained within the `Blob`. + +Then we can read from it, like this: + +```js +// get readableStream from blob +const readableStream = blob.stream(); +const stream = readableStream.getReader(); + +while (true) { + // for each iteration: value is the next blob fragment + let { done, value } = await stream.read(); + if (done) { + // no more data in the stream + console.log('all blob processed.'); + break; + } + + // do something with the data portion we've just read from the blob + console.log(value); +} +``` ## Summary @@ -231,9 +261,11 @@ While `ArrayBuffer`, `Uint8Array` and other `BufferSource` are "binary data", a That makes Blobs convenient for upload/download operations, that are so common in the browser. -Methods that perform web-requests, such as [XMLHttpRequest](info:xmlhttprequest), [fetch](info:fetch-basics) and so on, can work with `Blob` natively, as well as with other binary types. +Methods that perform web-requests, such as [XMLHttpRequest](info:xmlhttprequest), [fetch](info:fetch) and so on, can work with `Blob` natively, as well as with other binary types. + +We can easily convert between `Blob` and low-level binary data types: -We can easily convert betweeen `Blob` and low-level binary data types: +- We can make a `Blob` from a typed array using `new Blob(...)` constructor. +- We can get back `ArrayBuffer` from a Blob using `blob.arrayBuffer()`, and then create a view over it for low-level binary processing. -- We can make a Blob from a typed array using `new Blob(...)` constructor. -- We can get back `ArrayBuffer` from a Blob using `FileReader`, and then create a view over it for low-level binary processing. +Conversion streams are very useful when we need to handle large blob. You can easily create a `ReadableStream` from a blob. The `Blob` interface's `stream()` method returns a `ReadableStream` which upon reading returns the data contained within the blob. diff --git a/4-binary/04-file/article.md b/4-binary/04-file/article.md index af2671298..20878b650 100644 --- a/4-binary/04-file/article.md +++ b/4-binary/04-file/article.md @@ -1,6 +1,6 @@ # File and FileReader -A [File](https://www.w3.org/TR/FileAPI/#dfn-file) object inherits from `Blob`, but is extended with filesystem-related capabilities. +A [File](https://www.w3.org/TR/FileAPI/#dfn-file) object inherits from `Blob` and is extended with filesystem-related capabilities. There are two ways to obtain it. @@ -10,14 +10,18 @@ First, there's a constructor, similar to `Blob`: new File(fileParts, fileName, [options]) ``` -- **`fileParts`** -- is an array of Blob/BufferSource/String value, same as `Blob`. +- **`fileParts`** -- is an array of Blob/BufferSource/String values. - **`fileName`** -- file name string. - **`options`** -- optional object: - - **`lastModified`** -- a timestamp (integer date) of last modification. + - **`lastModified`** -- the timestamp (integer date) of last modification. -Second, more often we get a file from `<input type="file">` or drag'n'drop or other browser interfaces. Then the file gets these from OS. +Second, more often we get a file from `<input type="file">` or drag'n'drop or other browser interfaces. In that case, the file gets this information from OS. -For instance: +As `File` inherits from `Blob`, `File` objects have the same properties, plus: +- `name` -- the file name, +- `lastModified` -- the timestamp of last modification. + +That's how we can get a `File` object from `<input type="file">`: ```html run <input type="file" onchange="showFile(this)"> @@ -38,9 +42,9 @@ The input may select multiple files, so `input.files` is an array-like object wi ## FileReader -[FileReader](https://www.w3.org/TR/FileAPI/#dfn-filereader) is an object with the sole purpose of reading from `Blob` (and hence `File` too) objects. +[FileReader](https://www.w3.org/TR/FileAPI/#dfn-filereader) is an object with the sole purpose of reading data from `Blob` (and hence `File` too) objects. -It's event based, as reading from disk may take time. +It delivers the data using events, as reading from disk may take time. The constructor: @@ -50,17 +54,23 @@ let reader = new FileReader(); // no arguments The main methods: -- **`readAsArrayBuffer(blob)`** -- read the data as `ArrayBuffer` -- **`readAsText(blob, [encoding])`** -- read the data as a string (encoding is `utf-8` by default) -- **`readAsDataURL(blob)`** -- encode the data as base64 data url. +- **`readAsArrayBuffer(blob)`** -- read the data in binary format `ArrayBuffer`. +- **`readAsText(blob, [encoding])`** -- read the data as a text string with the given encoding (`utf-8` by default). +- **`readAsDataURL(blob)`** -- read the binary data and encode it as base64 data url. - **`abort()`** -- cancel the operation. +The choice of `read*` method depends on which format we prefer, how we're going to use the data. + +- `readAsArrayBuffer` -- for binary files, to do low-level binary operations. For high-level operations, like slicing, `File` inherits from `Blob`, so we can call them directly, without reading. +- `readAsText` -- for text files, when we'd like to get a string. +- `readAsDataURL` -- when we'd like to use this data in `src` for `img` or another tag. There's an alternative to reading a file for that, as discussed in chapter <info:blob>: `URL.createObjectURL(file)`. + As the reading proceeds, there are events: - `loadstart` -- loading started. - `progress` -- occurs during reading. - `load` -- no errors, reading complete. - `abort` -- `abort()` called. -- `error` -- error has occured. +- `error` -- error has occurred. - `loadend` -- reading finished with either success or failure. When the reading is finished, we can access the result as: @@ -95,28 +105,28 @@ function readFile(input) { ``` ```smart header="`FileReader` for blobs" -As mentioned in the chapter <info:blob>, `FileReader` works for any blobs, not just files. +As mentioned in the chapter <info:blob>, `FileReader` can read not just files, but any blobs. -So we can use it to convert a blob to another format: +We can use it to convert a blob to another format: - `readAsArrayBuffer(blob)` -- to `ArrayBuffer`, - `readAsText(blob, [encoding])` -- to string (an alternative to `TextDecoder`), - `readAsDataURL(blob)` -- to base64 data url. ``` -```smart header="`FileReaderSync` is available for workers only" +```smart header="`FileReaderSync` is available inside Web Workers" For Web Workers, there also exists a synchronous variant of `FileReader`, called [FileReaderSync](https://www.w3.org/TR/FileAPI/#FileReaderSync). Its reading methods `read*` do not generate events, but rather return a result, as regular functions do. -That's only inside a Web Worker though, because delays and hang-ups in Web Workers are less important, they do not affect the page. +That's only inside a Web Worker though, because delays in synchronous calls, that are possible while reading from files, in Web Workers are less important. They do not affect the page. ``` ## Summary `File` objects inherit from `Blob`. -In addition to `Blob` methods and properties, `File` objects also have `fileName` and `lastModified` properties, plus the internal ability to read from filesystem. We usually get `File` objects from user input, like `<input>` or drag'n'drop. +In addition to `Blob` methods and properties, `File` objects also have `name` and `lastModified` properties, plus the internal ability to read from filesystem. We usually get `File` objects from user input, like `<input>` or Drag'n'Drop events (`ondragend`). `FileReader` objects can read from a file or a blob, in one of three formats: - String (`readAsText`). @@ -125,4 +135,4 @@ In addition to `Blob` methods and properties, `File` objects also have `fileName In many cases though, we don't have to read the file contents. Just as we did with blobs, we can create a short url with `URL.createObjectURL(file)` and assign it to `<a>` or `<img>`. This way the file can be downloaded or shown up as an image, as a part of canvas etc. -And if we're going to send a `File` over a network, that's also easy, as network API like `XMLHttpRequest` or `fetch` natively accepts `File` objects. +And if we're going to send a `File` over a network, that's also easy: network API like `XMLHttpRequest` or `fetch` natively accepts `File` objects. diff --git a/5-network/01-fetch-basics/article.md b/5-network/01-fetch-basics/article.md deleted file mode 100644 index 783c39962..000000000 --- a/5-network/01-fetch-basics/article.md +++ /dev/null @@ -1,361 +0,0 @@ - -# Fetch: Basics - -Method `fetch()` is the modern way of sending requests over HTTP. - -It evolved for several years and continues to improve, right now the support is pretty solid among browsers. - -The basic syntax is: - -```js -let promise = fetch(url, [options]) -``` - -- **`url`** -- the URL to access. -- **`options`** -- optional parameters: method, headers etc. - -The browser starts the request right away and returns a `promise`. - -Getting a response is usually a two-stage process. - -**The `promise` resolves with an object of the built-in [Response](https://fetch.spec.whatwg.org/#response-class) class as soon as the server responds with headers.** - - -So we can check HTTP status, to see whether it is successful or not, check headers, but don't have the body yet. - -The promise rejects if the `fetch` was unable to make HTTP-request, e.g. network problems, or there's no such site. HTTP-errors, even such as 404 or 500, are considered a normal flow. - -We can see them in response properties: - -- **`ok`** -- boolean, `true` if the HTTP status code is 200-299. -- **`status`** -- HTTP status code. - -For example: - -```js -let response = await fetch(url); - -if (response.ok) { // if HTTP-status is 200-299 - // get the response body (see below) - let json = await response.json(); -} else { - alert("HTTP-Error: " + response.status); -} -``` - -To get the response body, we need to use an additional method call. - -`Response` provides multiple promise-based methods to access the body in various formats: - -- **`response.json()`** -- parse the response as JSON object, -- **`response.text()`** -- return the response as text, -- **`response.formData()`** -- return the response as FormData object (form/multipart encoding), -- **`response.blob()`** -- return the response as [Blob](info:blob) (binary data with type), -- **`response.arrayBuffer()`** -- return the response as [ArrayBuffer](info:arraybuffer-binary-arrays) (pure binary data), -- additionally, `response.body` is a [ReadableStream](https://streams.spec.whatwg.org/#rs-class) object, it allows to read the body chunk-by-chunk, we'll see an example later. - -For instance, here we get a JSON-object with latest commits from GitHub: - -```js run async -let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits'); - -*!* -let commits = await response.json(); // read response body and parse as JSON -*/!* - -alert(commits[0].author.login); -``` - -Or, the same using pure promises syntax: - -```js run -fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits') - .then(response => response.json()) - .then(commits => alert(commits[0].author.login)); -``` - -To get the text: -```js -let text = await response.text(); -``` - -And for the binary example, let's fetch and show an image (see chapter [Blob](info:blob) for details about operations on blobs): - -```js async run -let response = await fetch('/article/fetch/logo-fetch.svg'); - -*!* -let blob = await response.blob(); // download as Blob object -*/!* - -// create <img> for it -let img = document.createElement('img'); -img.style = 'position:fixed;top:10px;left:10px;width:100px'; -document.body.append(img); - -// show it -img.src = URL.createObjectURL(blob); - -setTimeout(() => { // hide after two seconds - img.remove(); - URL.revokeObjectURL(img.src); -}, 2000); -``` - -````warn -We can choose only one body-parsing method. - -If we got the response with `response.text()`, then `response.json()` won't work, as the body content has already been processed. - -```js -let text = await response.text(); // response body consumed -let parsed = await response.json(); // fails (already consumed) -```` - -## Headers - -There's a Map-like headers object in `response.headers`. - -We can get individual headers or iterate over them: - -```js run async -let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits'); - -// get one header -alert(response.headers.get('Content-Type')); // application/json; charset=utf-8 - -// iterate over all headers -for (let [key, value] of response.headers) { - alert(`${key} = ${value}`); -} -``` - -To set a header, we can use the `headers` option, like this: - -```js -let response = fetch(protectedUrl, { - headers: { - Authentication: 'abcdef' - } -}); -``` - -...But there's a list of [forbidden HTTP headers](https://fetch.spec.whatwg.org/#forbidden-header-name) that we can't set: - -- `Accept-Charset`, `Accept-Encoding` -- `Access-Control-Request-Headers` -- `Access-Control-Request-Method` -- `Connection` -- `Content-Length` -- `Cookie`, `Cookie2` -- `Date` -- `DNT` -- `Expect` -- `Host` -- `Keep-Alive` -- `Origin` -- `Referer` -- `TE` -- `Trailer` -- `Transfer-Encoding` -- `Upgrade` -- `Via` -- `Proxy-*` -- `Sec-*` - -These headers ensure proper and safe HTTP, so they are controlled exclusively by the browser. - -## POST requests - -To make a `POST` request, or a request with another method, we need to use `fetch` options: - -- **`method`** -- HTTP-method, e.g. `POST`, -- **`body`** -- one of: - - a string (e.g. JSON), - - `FormData` object, to submit the data as `form/multipart`, - - `Blob`/`BufferSource` to send binary data, - - [URLSearchParams](info:url), to submit the data as `x-www-form-urlencoded`, rarely used. - -Let's see examples. - -## Submit JSON - -This code submits a `user` object as JSON: - -```js run async -let user = { - name: 'John', - surname: 'Smith' -}; - -*!* -let response = await fetch('/article/fetch-basics/post/user', { - method: 'POST', - headers: { - 'Content-Type': 'application/json;charset=utf-8' - }, - body: JSON.stringify(user) -}); -*/!* - -let result = await response.json(); -alert(result.message); -``` - -Please note, if the body is a string, then `Content-Type` is set to `text/plain;charset=UTF-8` by default. So we use `headers` option to send `application/json` instead. - -## Submit a form - -Let's do the same with an HTML `<form>`. - - -```html run -<form id="formElem"> - <input type="text" name="name" value="John"> - <input type="text" name="surname" value="Smith"> -</form> - -<script> -(async () => { - let response = await fetch('/article/fetch-basics/post/user', { - method: 'POST', -*!* - body: new FormData(formElem) -*/!* - }); - - let result = await response.json(); - - alert(result.message); -})(); -</script> -``` - -Here [FormData](https://xhr.spec.whatwg.org/#formdata) automatically encodes the form, `<input type="file">` fields are handled also, and sends it using `Content-Type: form/multipart`. - -## Submit an image - -We can also submit binary data directly using `Blob` or `BufferSource`. - -For example, here's a `<canvas>` where we can draw by moving a mouse. A click on the "submit" button sends the image to server: - -```html run autorun height="90" -<body style="margin:0"> - <canvas id="canvasElem" width="100" height="80" style="border:1px solid"></canvas> - - <input type="button" value="Submit" onclick="submit()"> - - <script> - canvasElem.onmousemove = function(e) { - let ctx = canvasElem.getContext('2d'); - ctx.lineTo(e.clientX, e.clientY); - ctx.stroke(); - }; - - async function submit() { - let blob = await new Promise(resolve => canvasElem.toBlob(resolve, 'image/png')); - let response = await fetch('/article/fetch-basics/post/image', { - method: 'POST', - body: blob - }); - let result = await response.json(); - alert(result.message); - } - - </script> -</body> -``` - -Here we also didn't need to set `Content-Type` manually, because a `Blob` object has a built-in type (here `image/png`, as generated by `toBlob`). - -The `submit()` function can be rewritten without `async/await` like this: - -```js -function submit() { - canvasElem.toBlob(function(blob) { - fetch('/article/fetch-basics/post/image', { - method: 'POST', - body: blob - }) - .then(response => response.json()) - .then(result => alert(JSON.stringify(result, null, 2))) - }, 'image/png'); -} -``` - -## Custom FormData with image - -In practice though, it's often more convenient to send an image as a part of the form, with additional fields, such as "name" and other metadata. - -Also, servers are usually more suited to accept multipart-encoded forms, rather than raw binary data. - -```html run autorun height="90" -<body style="margin:0"> - <canvas id="canvasElem" width="100" height="80" style="border:1px solid"></canvas> - - <input type="button" value="Submit" onclick="submit()"> - - <script> - canvasElem.onmousemove = function(e) { - let ctx = canvasElem.getContext('2d'); - ctx.lineTo(e.clientX, e.clientY); - ctx.stroke(); - }; - - async function submit() { - let blob = await new Promise(resolve => canvasElem.toBlob(resolve, 'image/png')); - -*!* - let formData = new FormData(); - formData.append("name", "myImage"); - formData.append("image", blob); -*/!* - - let response = await fetch('/article/fetch-basics/post/image-form', { - method: 'POST', - body: formData - }); - let result = await response.json(); - alert(result.message); - } - - </script> -</body> -``` - -Now, from the server standpoint, the image is a "file" in the form. - -## Summary - -A typical fetch request consists of two `awaits`: - -```js -let response = await fetch(url, options); // resolves with response headers -let result = await response.json(); // read body as json -``` - -Or, promise-style: -```js -fetch(url, options) - .then(response => response.json()) - .then(result => /* process result */) -``` - -Response properties: -- `response.status` -- HTTP code of the response, -- `response.ok` -- `true` is the status is 200-299. -- `response.headers` -- Map-like object with HTTP headers. - -Methods to get response body: -- **`response.json()`** -- parse the response as JSON object, -- **`response.text()`** -- return the response as text, -- **`response.formData()`** -- return the response as FormData object (form/multipart encoding), -- **`response.blob()`** -- return the response as [Blob](info:blob) (binary data with type), -- **`response.arrayBuffer()`** -- return the response as [ArrayBuffer](info:arraybuffer-binary-arrays) (pure binary data), - -Fetch options so far: -- `method` -- HTTP-method, -- `headers` -- an object with request headers (not any header is allowed), -- `body` -- string/FormData/BufferSource/Blob/UrlSearchParams data to submit. - -In the next chapters we'll see more options and use cases. diff --git a/5-network/01-fetch/01-fetch-users/_js.view/solution.js b/5-network/01-fetch/01-fetch-users/_js.view/solution.js new file mode 100644 index 000000000..da448b47a --- /dev/null +++ b/5-network/01-fetch/01-fetch-users/_js.view/solution.js @@ -0,0 +1,24 @@ + +async function getUsers(names) { + let jobs = []; + + for(let name of names) { + let job = fetch(`https://api.github.com/users/${name}`).then( + successResponse => { + if (successResponse.status != 200) { + return null; + } else { + return successResponse.json(); + } + }, + failResponse => { + return null; + } + ); + jobs.push(job); + } + + let results = await Promise.all(jobs); + + return results; +} diff --git a/5-network/01-fetch/01-fetch-users/_js.view/source.js b/5-network/01-fetch/01-fetch-users/_js.view/source.js new file mode 100644 index 000000000..0c62e7bb5 --- /dev/null +++ b/5-network/01-fetch/01-fetch-users/_js.view/source.js @@ -0,0 +1,4 @@ + +async function getUsers(names) { + /* your code */ +} diff --git a/5-network/01-fetch/01-fetch-users/_js.view/test.js b/5-network/01-fetch/01-fetch-users/_js.view/test.js new file mode 100644 index 000000000..95eaf876e --- /dev/null +++ b/5-network/01-fetch/01-fetch-users/_js.view/test.js @@ -0,0 +1,10 @@ +describe("getUsers", function() { + + it("gets users from GitHub", async function() { + let users = await getUsers(['iliakan', 'remy', 'no.such.users']); + assert.equal(users[0].login, 'iliakan'); + assert.equal(users[1].login, 'remy'); + assert.equal(users[2], null); + }); + +}); diff --git a/5-network/01-fetch/01-fetch-users/solution.md b/5-network/01-fetch/01-fetch-users/solution.md new file mode 100644 index 000000000..3cb88e4ea --- /dev/null +++ b/5-network/01-fetch/01-fetch-users/solution.md @@ -0,0 +1,40 @@ + +To fetch a user we need: `fetch('https://api.github.com/users/USERNAME')`. + +If the response has status `200`, call `.json()` to read the JS object. + +Otherwise, if a `fetch` fails, or the response has non-200 status, we just return `null` in the resulting array. + +So here's the code: + +```js demo +async function getUsers(names) { + let jobs = []; + + for(let name of names) { + let job = fetch(`https://api.github.com/users/${name}`).then( + successResponse => { + if (successResponse.status != 200) { + return null; + } else { + return successResponse.json(); + } + }, + failResponse => { + return null; + } + ); + jobs.push(job); + } + + let results = await Promise.all(jobs); + + return results; +} +``` + +Please note: `.then` call is attached directly to `fetch`, so that when we have the response, it doesn't wait for other fetches, but starts to read `.json()` immediately. + +If we used `await Promise.all(names.map(name => fetch(...)))`, and call `.json()` on the results, then it would wait for all fetches to respond. By adding `.json()` directly to each `fetch`, we ensure that individual fetches start reading data as JSON without waiting for each other. + +That's an example of how low-level Promise API can still be useful even if we mainly use `async/await`. diff --git a/5-network/01-fetch/01-fetch-users/task.md b/5-network/01-fetch/01-fetch-users/task.md new file mode 100644 index 000000000..4605b4955 --- /dev/null +++ b/5-network/01-fetch/01-fetch-users/task.md @@ -0,0 +1,13 @@ +# Fetch users from GitHub + +Create an async function `getUsers(names)`, that gets an array of GitHub logins, fetches the users from GitHub and returns an array of GitHub users. + +The GitHub url with user information for the given `USERNAME` is: `https://api.github.com/users/USERNAME`. + +There's a test example in the sandbox. + +Important details: + +1. There should be one `fetch` request per user. +2. Requests shouldn't wait for each other. So that the data arrives as soon as possible. +3. If any request fails, or if there's no such user, the function should return `null` in the resulting array. diff --git a/5-network/01-fetch/article.md b/5-network/01-fetch/article.md new file mode 100644 index 000000000..4669fc451 --- /dev/null +++ b/5-network/01-fetch/article.md @@ -0,0 +1,316 @@ + +# Fetch + +JavaScript can send network requests to the server and load new information whenever it's needed. + +For example, we can use a network request to: + +- Submit an order, +- Load user information, +- Receive latest updates from the server, +- ...etc. + +...And all of that without reloading the page! + +There's an umbrella term "AJAX" (abbreviated <b>A</b>synchronous <b>J</b>avaScript <b>A</b>nd <b>X</b>ML) for network requests from JavaScript. We don't have to use XML though: the term comes from old times, that's why that word is there. You may have heard that term already. + +There are multiple ways to send a network request and get information from the server. + +The `fetch()` method is modern and versatile, so we'll start with it. It's not supported by old browsers (can be polyfilled), but very well supported among the modern ones. + +The basic syntax is: + +```js +let promise = fetch(url, [options]) +``` + +- **`url`** -- the URL to access. +- **`options`** -- optional parameters: method, headers etc. + +Without `options`, this is a simple GET request, downloading the contents of the `url`. + +The browser starts the request right away and returns a promise that the calling code should use to get the result. + +Getting a response is usually a two-stage process. + +**First, the `promise`, returned by `fetch`, resolves with an object of the built-in [Response](https://fetch.spec.whatwg.org/#response-class) class as soon as the server responds with headers.** + +At this stage we can check HTTP status, to see whether it is successful or not, check headers, but don't have the body yet. + +The promise rejects if the `fetch` was unable to make HTTP-request, e.g. network problems, or there's no such site. Abnormal HTTP-statuses, such as 404 or 500 do not cause an error. + +We can see HTTP-status in response properties: + +- **`status`** -- HTTP status code, e.g. 200. +- **`ok`** -- boolean, `true` if the HTTP status code is 200-299. + +For example: + +```js +let response = await fetch(url); + +if (response.ok) { // if HTTP-status is 200-299 + // get the response body (the method explained below) + let json = await response.json(); +} else { + alert("HTTP-Error: " + response.status); +} +``` + +**Second, to get the response body, we need to use an additional method call.** + +`Response` provides multiple promise-based methods to access the body in various formats: + +- **`response.text()`** -- read the response and return as text, +- **`response.json()`** -- parse the response as JSON, +- **`response.formData()`** -- return the response as `FormData` object (explained in the [next chapter](info:formdata)), +- **`response.blob()`** -- return the response as [Blob](info:blob) (binary data with type), +- **`response.arrayBuffer()`** -- return the response as [ArrayBuffer](info:arraybuffer-binary-arrays) (low-level representation of binary data), +- additionally, `response.body` is a [ReadableStream](https://streams.spec.whatwg.org/#rs-class) object, it allows you to read the body chunk-by-chunk, we'll see an example later. + +For instance, let's get a JSON-object with latest commits from GitHub: + +```js run async +let url = 'https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits'; +let response = await fetch(url); + +*!* +let commits = await response.json(); // read response body and parse as JSON +*/!* + +alert(commits[0].author.login); +``` + +Or, the same without `await`, using pure promises syntax: + +```js run +fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits') + .then(response => response.json()) + .then(commits => alert(commits[0].author.login)); +``` + +To get the response text, `await response.text()` instead of `.json()`: + +```js run async +let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits'); + +let text = await response.text(); // read response body as text + +alert(text.slice(0, 80) + '...'); +``` + +As a show-case for reading in binary format, let's fetch and show a logo image of ["fetch" specification](https://fetch.spec.whatwg.org) (see chapter [Blob](info:blob) for details about operations on `Blob`): + +```js async run +let response = await fetch('/article/fetch/logo-fetch.svg'); + +*!* +let blob = await response.blob(); // download as Blob object +*/!* + +// create <img> for it +let img = document.createElement('img'); +img.style = 'position:fixed;top:10px;left:10px;width:100px'; +document.body.append(img); + +// show it +img.src = URL.createObjectURL(blob); + +setTimeout(() => { // hide after three seconds + img.remove(); + URL.revokeObjectURL(img.src); +}, 3000); +``` + +````warn +We can choose only one body-reading method. + +If we've already got the response with `response.text()`, then `response.json()` won't work, as the body content has already been processed. + +```js +let text = await response.text(); // response body consumed +let parsed = await response.json(); // fails (already consumed) +``` +```` + +## Response headers + +The response headers are available in a Map-like headers object in `response.headers`. + +It's not exactly a Map, but it has similar methods to get individual headers by name or iterate over them: + +```js run async +let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits'); + +// get one header +alert(response.headers.get('Content-Type')); // application/json; charset=utf-8 + +// iterate over all headers +for (let [key, value] of response.headers) { + alert(`${key} = ${value}`); +} +``` + +## Request headers + +To set a request header in `fetch`, we can use the `headers` option. It has an object with outgoing headers, like this: + +```js +let response = fetch(protectedUrl, { + headers: { + Authentication: 'secret' + } +}); +``` + +...But there's a list of [forbidden HTTP headers](https://fetch.spec.whatwg.org/#forbidden-header-name) that we can't set: + +- `Accept-Charset`, `Accept-Encoding` +- `Access-Control-Request-Headers` +- `Access-Control-Request-Method` +- `Connection` +- `Content-Length` +- `Cookie`, `Cookie2` +- `Date` +- `DNT` +- `Expect` +- `Host` +- `Keep-Alive` +- `Origin` +- `Referer` +- `TE` +- `Trailer` +- `Transfer-Encoding` +- `Upgrade` +- `Via` +- `Proxy-*` +- `Sec-*` + +These headers ensure proper and safe HTTP, so they are controlled exclusively by the browser. + +## POST requests + +To make a `POST` request, or a request with another method, we need to use `fetch` options: + +- **`method`** -- HTTP-method, e.g. `POST`, +- **`body`** -- the request body, one of: + - a string (e.g. JSON-encoded), + - `FormData` object, to submit the data as `multipart/form-data`, + - `Blob`/`BufferSource` to send binary data, + - [URLSearchParams](info:url), to submit the data in `x-www-form-urlencoded` encoding, rarely used. + +The JSON format is used most of the time. + +For example, this code submits `user` object as JSON: + +```js run async +let user = { + name: 'John', + surname: 'Smith' +}; + +*!* +let response = await fetch('/article/fetch/post/user', { + method: 'POST', + headers: { + 'Content-Type': 'application/json;charset=utf-8' + }, + body: JSON.stringify(user) +}); +*/!* + +let result = await response.json(); +alert(result.message); +``` + +Please note, if the request `body` is a string, then `Content-Type` header is set to `text/plain;charset=UTF-8` by default. + +But, as we're going to send JSON, we use `headers` option to send `application/json` instead, the correct `Content-Type` for JSON-encoded data. + +## Sending an image + +We can also submit binary data with `fetch` using `Blob` or `BufferSource` objects. + +In this example, there's a `<canvas>` where we can draw by moving a mouse over it. A click on the "submit" button sends the image to the server: + +```html run autorun height="90" +<body style="margin:0"> + <canvas id="canvasElem" width="100" height="80" style="border:1px solid"></canvas> + + <input type="button" value="Submit" onclick="submit()"> + + <script> + canvasElem.onmousemove = function(e) { + let ctx = canvasElem.getContext('2d'); + ctx.lineTo(e.clientX, e.clientY); + ctx.stroke(); + }; + + async function submit() { + let blob = await new Promise(resolve => canvasElem.toBlob(resolve, 'image/png')); + let response = await fetch('/article/fetch/post/image', { + method: 'POST', + body: blob + }); + + // the server responds with confirmation and the image size + let result = await response.json(); + alert(result.message); + } + + </script> +</body> +``` + +Please note, here we don't set `Content-Type` header manually, because a `Blob` object has a built-in type (here `image/png`, as generated by `toBlob`). For `Blob` objects that type becomes the value of `Content-Type`. + +The `submit()` function can be rewritten without `async/await` like this: + +```js +function submit() { + canvasElem.toBlob(function(blob) { + fetch('/article/fetch/post/image', { + method: 'POST', + body: blob + }) + .then(response => response.json()) + .then(result => alert(JSON.stringify(result, null, 2))) + }, 'image/png'); +} +``` + +## Summary + +A typical fetch request consists of two `await` calls: + +```js +let response = await fetch(url, options); // resolves with response headers +let result = await response.json(); // read body as json +``` + +Or, without `await`: + +```js +fetch(url, options) + .then(response => response.json()) + .then(result => /* process result */) +``` + +Response properties: +- `response.status` -- HTTP code of the response, +- `response.ok` -- `true` if the status is 200-299. +- `response.headers` -- Map-like object with HTTP headers. + +Methods to get response body: +- **`response.text()`** -- return the response as text, +- **`response.json()`** -- parse the response as JSON object, +- **`response.formData()`** -- return the response as `FormData` object (`multipart/form-data` encoding, see the next chapter), +- **`response.blob()`** -- return the response as [Blob](info:blob) (binary data with type), +- **`response.arrayBuffer()`** -- return the response as [ArrayBuffer](info:arraybuffer-binary-arrays) (low-level binary data), + +Fetch options so far: +- `method` -- HTTP-method, +- `headers` -- an object with request headers (not any header is allowed), +- `body` -- the data to send (request body) as `string`, `FormData`, `BufferSource`, `Blob` or `UrlSearchParams` object. + +In the next chapters we'll see more options and use cases of `fetch`. diff --git a/5-network/01-fetch-basics/logo-fetch.svg b/5-network/01-fetch/logo-fetch.svg similarity index 100% rename from 5-network/01-fetch-basics/logo-fetch.svg rename to 5-network/01-fetch/logo-fetch.svg diff --git a/5-network/01-fetch-basics/post.view/server.js b/5-network/01-fetch/post.view/server.js similarity index 55% rename from 5-network/01-fetch-basics/post.view/server.js rename to 5-network/01-fetch/post.view/server.js index 18610676a..b55870b2a 100644 --- a/5-network/01-fetch-basics/post.view/server.js +++ b/5-network/01-fetch/post.view/server.js @@ -2,7 +2,6 @@ const Koa = require('koa'); const app = new Koa(); const bodyParser = require('koa-bodyparser'); const getRawBody = require('raw-body') -const busboy = require('async-busboy'); const Router = require('koa-router'); let router = new Router(); @@ -22,28 +21,6 @@ router.post('/image', async (ctx) => { }; }); -router.post('/image-form', async (ctx) => { - - let files = []; - const { fields } = await busboy(ctx.req, { - onFile(fieldname, file, filename, encoding, mimetype) { - // read all file stream to continue - getRawBody(file, { limit: '1mb'}).then(body => { - files.push({ - fieldname, - filename, - length: body.length - }); - }) - - } - }); - - ctx.body = { - message: `Image saved, name: ${fields.name}, size:${files[0].length}.` - }; -}); - app .use(bodyParser()) .use(router.routes()) diff --git a/5-network/02-fetch-progress/article.md b/5-network/02-fetch-progress/article.md deleted file mode 100644 index e7e945f66..000000000 --- a/5-network/02-fetch-progress/article.md +++ /dev/null @@ -1,104 +0,0 @@ - -# Fetch: Download progress - -Fetch allows to track download progress, but not upload progress. - -Please note: there's currently no way for fetch to track upload progress. For that purpose, please use [XMLHttpRequest](info:xmlhttprequest). - -To track download progress, we can use `response.body` property. It's a "readable stream" -- a special object that provides body chunk-by-chunk, as it comes, so we can see how much is available at the moment. - -Here's the sketch of code that uses it to read response: - -```js -// instead of response.json() and other methods -const reader = response.body.getReader(); - -// infinite loop while the body is downloading -while(true) { - // done is true for the last chunk - // value is Uint8Array of the chunk bytes - const {done, value} = await reader.read(); - - if (done) { - break; - } - - console.log(`Received ${value.length} bytes`) -} -``` - -So, we loop, while `await reader.read()` returns response chunks. - -A chunk has two properties: -- **`done`** -- true when the reading is complete. -- **`value`** -- a typed array of bytes: `Uint8Array`. - -To log the progress, we just need to count chunks. - -Here's the full code to get response and log the progress, more explanations follow: - -```js run async -// Step 1: start the fetch and obtain a reader -let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits?per_page=100'); - -const reader = response.body.getReader(); - -// Step 2: get total length -const contentLength = +response.headers.get('Content-Length'); - -// Step 3: read the data -let receivedLength = 0; // length at the moment -let chunks = []; // array of received binary chunks (comprises the body) -while(true) { - const {done, value} = await reader.read(); - - if (done) { - break; - } - - chunks.push(value); - receivedLength += value.length; - - console.log(`Received ${receivedLength} of ${contentLength}`) -} - -// Step 4: concatenate chunks into single Uint8Array -let chunksAll = new Uint8Array(receivedLength); // (4.1) -let position = 0; -for(let chunk of chunks) { - chunksAll.set(chunk, position); // (4.2) - position += chunk.length; -} - -// Step 5: decode into a string -let result = new TextDecoder("utf-8").decode(chunksAll); - -// We're done! -let commits = JSON.parse(result); -alert(commits[0].author.login); -``` - -Let's explain that step-by-step: - -1. We perform `fetch` as usual, but instead of calling `response.json()`, we obtain a stream reader `response.body.getReader()`. - - Please note, we can't use both these methods to read the same response. Either use a reader or a response method to get the result. -2. Prior to reading, we can figure out the full response length from the `Content-Length` header. - - It may be absent for cross-domain requests (see chapter <info:fetch-crossorigin>) and, well, technically a server doesn't have to set it. But usually it's at place. -3. Call `await reader.read()` until it's done. - - We gather response `chunks` in the array. That's important, because after the response is consumed, we won't be able to "re-read" it using `response.json()` or another way (you can try, there'll be an error). -4. At the end, we have `chunks` -- an array of `Uint8Array` byte chunks. We need to join them into a single result. Unfortunately, there's no single method that concatenates those, so there's some code to do that: - 1. We create `new Uint8Array(receivedLength)` -- a same-typed array with the combined length. - 2. Then use `.set(chunk, position)` method to copy each `chunk` one after another in the resulting array. -5. We have the result in `chunksAll`. It's a byte array though, not a string. - - To create a string, we need to interpret these bytes. The built-in [TextDecoder](info:text-decoder) does exactly that. Then we can `JSON.parse` it. - -What if we need binary content instead of JSON? That's even simpler. Instead of steps 4 and 5, we could make a blob of all chunks: -```js -let blob = new Blob(chunks); -``` - -Once again, please note, that's not for upload progress (no way now), only for download progress. diff --git a/5-network/02-formdata/article.md b/5-network/02-formdata/article.md new file mode 100644 index 000000000..a73d554b1 --- /dev/null +++ b/5-network/02-formdata/article.md @@ -0,0 +1,189 @@ + +# FormData + +This chapter is about sending HTML forms: with or without files, with additional fields and so on. + +[FormData](https://xhr.spec.whatwg.org/#interface-formdata) objects can help with that. As you might have guessed, it's the object to represent HTML form data. + +The constructor is: +```js +let formData = new FormData([form]); +``` + +If HTML `form` element is provided, it automatically captures its fields. + +The special thing about `FormData` is that network methods, such as `fetch`, can accept a `FormData` object as a body. It's encoded and sent out with `Content-Type: multipart/form-data`. + +From the server point of view, that looks like a usual form submission. + +## Sending a simple form + +Let's send a simple form first. + +As you can see, that's almost one-liner: + +```html run autorun +<form id="formElem"> + <input type="text" name="name" value="John"> + <input type="text" name="surname" value="Smith"> + <input type="submit"> +</form> + +<script> + formElem.onsubmit = async (e) => { + e.preventDefault(); + + let response = await fetch('/article/formdata/post/user', { + method: 'POST', +*!* + body: new FormData(formElem) +*/!* + }); + + let result = await response.json(); + + alert(result.message); + }; +</script> +``` + +In this example, the server code is not presented, as it's beyond our scope. The server accepts the POST request and replies "User saved". + +## FormData Methods + +We can modify fields in `FormData` with methods: + +- `formData.append(name, value)` - add a form field with the given `name` and `value`, +- `formData.append(name, blob, fileName)` - add a field as if it were `<input type="file">`, the third argument `fileName` sets file name (not form field name), as it were a name of the file in user's filesystem, +- `formData.delete(name)` - remove the field with the given `name`, +- `formData.get(name)` - get the value of the field with the given `name`, +- `formData.has(name)` - if there exists a field with the given `name`, returns `true`, otherwise `false` + +A form is technically allowed to have many fields with the same `name`, so multiple calls to `append` add more same-named fields. + +There's also method `set`, with the same syntax as `append`. The difference is that `.set` removes all fields with the given `name`, and then appends a new field. So it makes sure there's only one field with such `name`, the rest is just like `append`: + +- `formData.set(name, value)`, +- `formData.set(name, blob, fileName)`. + +Also we can iterate over formData fields using `for..of` loop: + +```js run +let formData = new FormData(); +formData.append('key1', 'value1'); +formData.append('key2', 'value2'); + +// List key/value pairs +for(let [name, value] of formData) { + alert(`${name} = ${value}`); // key1 = value1, then key2 = value2 +} +``` + +## Sending a form with a file + +The form is always sent as `Content-Type: multipart/form-data`, this encoding allows to send files. So, `<input type="file">` fields are sent also, similar to a usual form submission. + +Here's an example with such form: + +```html run autorun +<form id="formElem"> + <input type="text" name="firstName" value="John"> + Picture: <input type="file" name="picture" accept="image/*"> + <input type="submit"> +</form> + +<script> + formElem.onsubmit = async (e) => { + e.preventDefault(); + + let response = await fetch('/article/formdata/post/user-avatar', { + method: 'POST', +*!* + body: new FormData(formElem) +*/!* + }); + + let result = await response.json(); + + alert(result.message); + }; +</script> +``` + +## Sending a form with Blob data + +As we've seen in the chapter <info:fetch>, it's easy to send dynamically generated binary data e.g. an image, as `Blob`. We can supply it directly as `fetch` parameter `body`. + +In practice though, it's often convenient to send an image not separately, but as a part of the form, with additional fields, such as "name" and other metadata. + +Also, servers are usually more suited to accept multipart-encoded forms, rather than raw binary data. + +This example submits an image from `<canvas>`, along with some other fields, as a form, using `FormData`: + +```html run autorun height="90" +<body style="margin:0"> + <canvas id="canvasElem" width="100" height="80" style="border:1px solid"></canvas> + + <input type="button" value="Submit" onclick="submit()"> + + <script> + canvasElem.onmousemove = function(e) { + let ctx = canvasElem.getContext('2d'); + ctx.lineTo(e.clientX, e.clientY); + ctx.stroke(); + }; + + async function submit() { + let imageBlob = await new Promise(resolve => canvasElem.toBlob(resolve, 'image/png')); + +*!* + let formData = new FormData(); + formData.append("firstName", "John"); + formData.append("image", imageBlob, "image.png"); +*/!* + + let response = await fetch('/article/formdata/post/image-form', { + method: 'POST', + body: formData + }); + let result = await response.json(); + alert(result.message); + } + + </script> +</body> +``` + +Please note how the image `Blob` is added: + +```js +formData.append("image", imageBlob, "image.png"); +``` + +That's same as if there were `<input type="file" name="image">` in the form, and the visitor submitted a file named `"image.png"` (3rd argument) with the data `imageBlob` (2nd argument) from their filesystem. + +The server reads form data and the file, as if it were a regular form submission. + +## Summary + +[FormData](https://xhr.spec.whatwg.org/#interface-formdata) objects are used to capture HTML form and submit it using `fetch` or another network method. + +We can either create `new FormData(form)` from an HTML form, or create an object without a form at all, and then append fields with methods: + +- `formData.append(name, value)` +- `formData.append(name, blob, fileName)` +- `formData.set(name, value)` +- `formData.set(name, blob, fileName)` + +Let's note two peculiarities here: + +1. The `set` method removes fields with the same name, `append` doesn't. That's the only difference between them. +2. To send a file, 3-argument syntax is needed, the last argument is a file name, that normally is taken from user filesystem for `<input type="file">`. + +Other methods are: + +- `formData.delete(name)` +- `formData.get(name)` +- `formData.has(name)` + +That's it! diff --git a/5-network/02-formdata/post.view/server.js b/5-network/02-formdata/post.view/server.js new file mode 100644 index 000000000..a335b07b1 --- /dev/null +++ b/5-network/02-formdata/post.view/server.js @@ -0,0 +1,78 @@ +const Koa = require('koa'); +const app = new Koa(); +const bodyParser = require('koa-bodyparser'); +const getRawBody = require('raw-body') +const busboy = require('async-busboy'); +const Router = require('koa-router'); + +let router = new Router(); + +router.post('/user', async (ctx) => { + ctx.body = { + message: "User saved" + }; +}); + +router.post('/image-form', async (ctx) => { + + let files = []; + const { fields } = await busboy(ctx.req, { + onFile(fieldname, file, filename, encoding, mimetype) { + // read all file stream to continue + let length = 0; + file.on('data', function(data) { + length += data.length; + }); + file.on('end', () => { + files.push({ + fieldname, + filename, + length + }); + }); + } + }); + + ctx.body = { + message: `Image saved, firstName: ${fields.firstName}, Image size:${files[0].length}, fileName: ${files[0].filename}` + }; +}); + + +router.post('/user-avatar', async (ctx) => { + + let files = []; + const { fields } = await busboy(ctx.req, { + onFile(fieldname, file, filename, encoding, mimetype) { + // read all file stream to continue + let length = 0; + file.on('data', function(data) { + length += data.length; + }); + file.on('end', () => { + files.push({ + fieldname, + filename, + length + }); + }); + + } + }); + + ctx.body = { + message: `User with picture, firstName: ${fields.firstName}, picture size:${files[0].length}` + }; +}); + +app + .use(bodyParser()) + .use(router.routes()) + .use(router.allowedMethods()); + + +if (!module.parent) { + http.createServer(app.callback()).listen(8080); +} else { + exports.accept = app.callback(); +} diff --git a/5-network/03-fetch-abort/article.md b/5-network/03-fetch-abort/article.md deleted file mode 100644 index 3e6f4cb42..000000000 --- a/5-network/03-fetch-abort/article.md +++ /dev/null @@ -1,110 +0,0 @@ - -# Fetch: Abort - -Aborting a `fetch` is a little bit tricky. Remember, `fetch` returns a promise. And JavaScript generally has no concept of "aborting" a promise. So how can we cancel a fetch? - -There's a special built-in object for such purposes: `AbortController`. - -The usage is pretty simple: - -- Step 1: create a controller: - - ```js - let controller = new AbortController(); - ``` - - A controller is an extremely simple object. It has a single method `abort()`, and a single property `signal`. When `abort()` is called, the `abort` event triggers on `controller.signal`: - - Like this: - - ```js run - let controller = new AbortController(); - let signal = controller.signal; - - // triggers when controller.abort() is called - signal.addEventListener('abort', () => alert("abort!")); - - controller.abort(); // abort! - - alert(signal.aborted); // true (after abort) - ``` - -- Step 2: pass the `signal` property to `fetch` option: - - ```js - let controller = new AbortController(); - fetch(url, { - signal: controller.signal - }); - ``` - - Now `fetch` listens to the signal. - -- Step 3: to abort, call `controller.abort()`: - - ```js - controller.abort(); - ``` - - We're done: `fetch` gets the event from `signal` and aborts the request. - -When a fetch is aborted, its promise rejects with an error named `AbortError`, so we should handle it: - -```js run async -// abort in 1 second -let controller = new AbortController(); -setTimeout(() => controller.abort(), 1000); - -try { - let response = await fetch('/article/fetch-abort/demo/hang', { - signal: controller.signal - }); -} catch(err) { - if (err.name == 'AbortError') { // handle abort() - alert("Aborted!"); - } else { - throw err; - } -} -``` - -**`AbortController` is scalable, it allows to cancel multiple fetches at once.** - -For instance, here we fetch many `urls` in parallel, and the controller aborts them all: - -```js -let urls = [...]; // a list of urls to fetch in parallel - -let controller = new AbortController(); - -let fetchJobs = urls.map(url => fetch(url, { - signal: controller.signal -})); - -let results = await Promise.all(fetchJobs); - -// from elsewhere: -// controller.abort() stops all fetches -``` - -If we have our own jobs, different from `fetch`, we can use a single `AbortController` to stop those, together with fetches. - - -```js -let urls = [...]; -let controller = new AbortController(); - -let ourJob = new Promise((resolve, reject) => { - ... - controller.signal.addEventListener('abort', reject); -}); - -let fetchJobs = urls.map(url => fetch(url, { - signal: controller.signal -})); - -let results = await Promise.all([...fetchJobs, ourJob]); - -// from elsewhere: -// controller.abort() stops all fetches and ourJob -``` diff --git a/5-network/03-fetch-progress/article.md b/5-network/03-fetch-progress/article.md new file mode 100644 index 000000000..76b05d514 --- /dev/null +++ b/5-network/03-fetch-progress/article.md @@ -0,0 +1,114 @@ + +# Fetch: Download progress + +The `fetch` method allows to track *download* progress. + +Please note: there's currently no way for `fetch` to track *upload* progress. For that purpose, please use [XMLHttpRequest](info:xmlhttprequest), we'll cover it later. + +To track download progress, we can use `response.body` property. It's a `ReadableStream` -- a special object that provides body chunk-by-chunk, as it comes. Readable streams are described in the [Streams API](https://streams.spec.whatwg.org/#rs-class) specification. + +Unlike `response.text()`, `response.json()` and other methods, `response.body` gives full control over the reading process, and we can count how much is consumed at any moment. + +Here's the sketch of code that reads the response from `response.body`: + +```js +// instead of response.json() and other methods +const reader = response.body.getReader(); + +// infinite loop while the body is downloading +while(true) { + // done is true for the last chunk + // value is Uint8Array of the chunk bytes + const {done, value} = await reader.read(); + + if (done) { + break; + } + + console.log(`Received ${value.length} bytes`) +} +``` + +The result of `await reader.read()` call is an object with two properties: +- **`done`** -- `true` when the reading is complete, otherwise `false`. +- **`value`** -- a typed array of bytes: `Uint8Array`. + +```smart +Streams API also describes asynchronous iteration over `ReadableStream` with `for await..of` loop, but it's not yet widely supported (see [browser issues](https://github.com/whatwg/streams/issues/778#issuecomment-461341033)), so we use `while` loop. +``` + +We receive response chunks in the loop, until the loading finishes, that is: until `done` becomes `true`. + +To log the progress, we just need for every received fragment `value` to add its length to the counter. + +Here's the full working example that gets the response and logs the progress in console, more explanations to follow: + +```js run async +// Step 1: start the fetch and obtain a reader +let response = await fetch('https://api.github.com/repos/javascript-tutorial/en.javascript.info/commits?per_page=100'); + +const reader = response.body.getReader(); + +// Step 2: get total length +const contentLength = +response.headers.get('Content-Length'); + +// Step 3: read the data +let receivedLength = 0; // received that many bytes at the moment +let chunks = []; // array of received binary chunks (comprises the body) +while(true) { + const {done, value} = await reader.read(); + + if (done) { + break; + } + + chunks.push(value); + receivedLength += value.length; + + console.log(`Received ${receivedLength} of ${contentLength}`) +} + +// Step 4: concatenate chunks into single Uint8Array +let chunksAll = new Uint8Array(receivedLength); // (4.1) +let position = 0; +for(let chunk of chunks) { + chunksAll.set(chunk, position); // (4.2) + position += chunk.length; +} + +// Step 5: decode into a string +let result = new TextDecoder("utf-8").decode(chunksAll); + +// We're done! +let commits = JSON.parse(result); +alert(commits[0].author.login); +``` + +Let's explain that step-by-step: + +1. We perform `fetch` as usual, but instead of calling `response.json()`, we obtain a stream reader `response.body.getReader()`. + + Please note, we can't use both these methods to read the same response: either use a reader or a response method to get the result. +2. Prior to reading, we can figure out the full response length from the `Content-Length` header. + + It may be absent for cross-origin requests (see chapter <info:fetch-crossorigin>) and, well, technically a server doesn't have to set it. But usually it's at place. +3. Call `await reader.read()` until it's done. + + We gather response chunks in the array `chunks`. That's important, because after the response is consumed, we won't be able to "re-read" it using `response.json()` or another way (you can try, there'll be an error). +4. At the end, we have `chunks` -- an array of `Uint8Array` byte chunks. We need to join them into a single result. Unfortunately, there's no single method that concatenates those, so there's some code to do that: + 1. We create `chunksAll = new Uint8Array(receivedLength)` -- a same-typed array with the combined length. + 2. Then use `.set(chunk, position)` method to copy each `chunk` one after another in it. +5. We have the result in `chunksAll`. It's a byte array though, not a string. + + To create a string, we need to interpret these bytes. The built-in [TextDecoder](info:text-decoder) does exactly that. Then we can `JSON.parse` it, if necessary. + + What if we need binary content instead of a string? That's even simpler. Replace steps 4 and 5 with a single line that creates a `Blob` from all chunks: + ```js + let blob = new Blob(chunks); + ``` + +At the end we have the result (as a string or a blob, whatever is convenient), and progress-tracking in the process. + +Once again, please note, that's not for *upload* progress (no way now with `fetch`), only for *download* progress. + +Also, if the size is unknown, we should check `receivedLength` in the loop and break it once it reaches a certain limit. So that the `chunks` won't overflow the memory. diff --git a/5-network/02-fetch-progress/logo-fetch.svg b/5-network/03-fetch-progress/logo-fetch.svg similarity index 100% rename from 5-network/02-fetch-progress/logo-fetch.svg rename to 5-network/03-fetch-progress/logo-fetch.svg diff --git a/5-network/02-fetch-progress/progress.view/index.html b/5-network/03-fetch-progress/progress.view/index.html similarity index 100% rename from 5-network/02-fetch-progress/progress.view/index.html rename to 5-network/03-fetch-progress/progress.view/index.html diff --git a/5-network/02-fetch-progress/progress.view/long.txt b/5-network/03-fetch-progress/progress.view/long.txt similarity index 100% rename from 5-network/02-fetch-progress/progress.view/long.txt rename to 5-network/03-fetch-progress/progress.view/long.txt diff --git a/5-network/04-fetch-abort/article.md b/5-network/04-fetch-abort/article.md new file mode 100644 index 000000000..eadc5aac2 --- /dev/null +++ b/5-network/04-fetch-abort/article.md @@ -0,0 +1,148 @@ + +# Fetch: Abort + +As we know, `fetch` returns a promise. And JavaScript generally has no concept of "aborting" a promise. So how can we cancel an ongoing `fetch`? E.g. if the user actions on our site indicate that the `fetch` isn't needed any more. + +There's a special built-in object for such purposes: `AbortController`. It can be used to abort not only `fetch`, but other asynchronous tasks as well. + +The usage is very straightforward: + +## The AbortController object + +Create a controller: + +```js +let controller = new AbortController(); +``` + +A controller is an extremely simple object. + +- It has a single method `abort()`, +- And a single property `signal` that allows to set event listeners on it. + +When `abort()` is called: +- `controller.signal` emits the `"abort"` event. +- `controller.signal.aborted` property becomes `true`. + +Generally, we have two parties in the process: +1. The one that performs a cancelable operation, it sets a listener on `controller.signal`. +2. The one that cancels: it calls `controller.abort()` when needed. + +Here's the full example (without `fetch` yet): + +```js run +let controller = new AbortController(); +let signal = controller.signal; + +// The party that performs a cancelable operation +// gets the "signal" object +// and sets the listener to trigger when controller.abort() is called +signal.addEventListener('abort', () => alert("abort!")); + +// The other party, that cancels (at any point later): +controller.abort(); // abort! + +// The event triggers and signal.aborted becomes true +alert(signal.aborted); // true +``` + +As we can see, `AbortController` is just a mean to pass `abort` events when `abort()` is called on it. + +We could implement the same kind of event listening in our code on our own, without the `AbortController` object. + +But what's valuable is that `fetch` knows how to work with the `AbortController` object. It's integrated in it. + +## Using with fetch + +To be able to cancel `fetch`, pass the `signal` property of an `AbortController` as a `fetch` option: + +```js +let controller = new AbortController(); +fetch(url, { + signal: controller.signal +}); +``` + +The `fetch` method knows how to work with `AbortController`. It will listen to `abort` events on `signal`. + +Now, to abort, call `controller.abort()`: + +```js +controller.abort(); +``` + +We're done: `fetch` gets the event from `signal` and aborts the request. + +When a fetch is aborted, its promise rejects with an error `AbortError`, so we should handle it, e.g. in `try..catch`. + +Here's the full example with `fetch` aborted after 1 second: + +```js run async +// abort in 1 second +let controller = new AbortController(); +setTimeout(() => controller.abort(), 1000); + +try { + let response = await fetch('/article/fetch-abort/demo/hang', { + signal: controller.signal + }); +} catch(err) { + if (err.name == 'AbortError') { // handle abort() + alert("Aborted!"); + } else { + throw err; + } +} +``` + +## AbortController is scalable + +`AbortController` is scalable. It allows to cancel multiple fetches at once. + +Here's a sketch of code that fetches many `urls` in parallel, and uses a single controller to abort them all: + +```js +let urls = [...]; // a list of urls to fetch in parallel + +let controller = new AbortController(); + +// an array of fetch promises +let fetchJobs = urls.map(url => fetch(url, { + signal: controller.signal +})); + +let results = await Promise.all(fetchJobs); + +// if controller.abort() is called from anywhere, +// it aborts all fetches +``` + +If we have our own asynchronous tasks, different from `fetch`, we can use a single `AbortController` to stop those, together with fetches. + +We just need to listen to its `abort` event in our tasks: + +```js +let urls = [...]; +let controller = new AbortController(); + +let ourJob = new Promise((resolve, reject) => { // our task + ... + controller.signal.addEventListener('abort', reject); +}); + +let fetchJobs = urls.map(url => fetch(url, { // fetches + signal: controller.signal +})); + +// Wait for fetches and our task in parallel +let results = await Promise.all([...fetchJobs, ourJob]); + +// if controller.abort() is called from anywhere, +// it aborts all fetches and ourJob +``` + +## Summary + +- `AbortController` is a simple object that generates an `abort` event on its `signal` property when the `abort()` method is called (and also sets `signal.aborted` to `true`). +- `fetch` integrates with it: we pass the `signal` property as the option, and then `fetch` listens to it, so it's possible to abort the `fetch`. +- We can use `AbortController` in our code. The "call `abort()`" -> "listen to `abort` event" interaction is simple and universal. We can use it even without `fetch`. diff --git a/5-network/03-fetch-abort/demo.view/server.js b/5-network/04-fetch-abort/demo.view/server.js similarity index 100% rename from 5-network/03-fetch-abort/demo.view/server.js rename to 5-network/04-fetch-abort/demo.view/server.js diff --git a/5-network/04-fetch-crossorigin/demo.view/index.html b/5-network/04-fetch-crossorigin/demo.view/index.html deleted file mode 100644 index 7e367598c..000000000 --- a/5-network/04-fetch-crossorigin/demo.view/index.html +++ /dev/null @@ -1,31 +0,0 @@ -<!doctype html> -<script> - -async function init() { - const response = await fetch('long.txt'); - const reader = response.body.getReader(); - - const contentLength = +response.headers.get('Content-Length'); - let receivedLength = 0; - - while(true) { - const chunk = await reader.read(); - - if (chunk.done) { - console.log("done!"); - break; - } - - receivedLength += chunk.value.length; - console.log(`${receivedLength}/${contentLength} received`) - } - - let result = await response.text(); - console.log(result); - //const chunkCount = await read(reader); - //console.log(`Finished! Received ${chunkCount} chunks.`); -} - -init(); - -</script> diff --git a/5-network/04-fetch-crossorigin/demo.view/server.js b/5-network/04-fetch-crossorigin/demo.view/server.js deleted file mode 100644 index 588ca0653..000000000 --- a/5-network/04-fetch-crossorigin/demo.view/server.js +++ /dev/null @@ -1,31 +0,0 @@ -const Koa = require('koa'); -const app = new Koa(); - -const Router = require('koa-router'); - -let router = new Router(); - -router.get('/script', async (ctx) => { - let callback = ctx.query.callback; - - if (!callback) { - ctx.throw(400, 'Callback required!'); - } - - ctx.type = 'application/javascript'; - ctx.body = `${callback}({ - temperature: 25, - humidity: 78 - })`; -}); - -app - .use(router.routes()) - .use(router.allowedMethods()); - - -if (!module.parent) { - http.createServer(app.callback()).listen(8080); -} else { - exports.accept = app.callback(); -} diff --git a/5-network/05-fetch-api/article.md b/5-network/05-fetch-api/article.md deleted file mode 100644 index 7b0330f26..000000000 --- a/5-network/05-fetch-api/article.md +++ /dev/null @@ -1,190 +0,0 @@ - -# Fetch API - -So far, we know quite a bit about fetch. - -Now let's see the rest of API, to cover all its abilities. - -Here's the full list of all possible fetch options with their default values (alternatives in comments): - -```js -let promise = fetch(url, { - method: "GET", // POST, PUT, DELETE, etc. - headers: { - "Content-Type": "text/plain;charset=UTF-8" // for a string body, depends on body - }, - body: undefined // string, FormData, Blob, BufferSource, or URLSearchParams - referrer: "about:client", // "" for no-referrer, or an url from the current origin - referrerPolicy: "no-referrer-when-downgrade", // no-referrer, origin, same-origin... - mode: "cors", // same-origin, no-cors - credentials: "same-origin", // omit, include - cache: "default", // no-store, reload, no-cache, force-cache, or only-if-cached - redirect: "follow", // manual, error - integrity: "", // a hash, like "sha256-abcdef1234567890" - keepalive: false, // true - signal: undefined, // AbortController to abort request - window: window // null -}); -``` - -An impressive list, right? - -We fully covered `method`, `headers` and `body` in the chapter <info:fetch-basics>. - -The `signal` option is covered in <info:fetch-abort>. - -Now let's explore the rest of options. - -## referrer, referrerPolicy - -These options govern how `fetch` sets HTTP `Referer` header. - -That header contains the url of the page that made the request. In most scenarios, it plays a very minor informational role, but sometimes, for security purposes, it makes sense to remove or modify it. -. - -**The `referrer` option allows to set any `Referer` within the current origin) or disable it.** - -To send no referer, set an empty string: -```js -fetch('/page', { -*!* - referrer: "" // no Referer header -*/!* -}); -``` - -To set another url within the current origin: - -```js -fetch('/page', { - // assuming we're on https://javascript.info - // we can set any Referer header, but only within the current origin -*!* - referrer: "https://javascript.info/anotherpage" -*/!* -}); -``` - -**The `referrerPolicy` option sets general rules for `Referer`.** - -Possible values are described in the [Referrer Policy specification](https://w3c.github.io/webappsec-referrer-policy/): - -- **`"no-referrer-when-downgrade"`** -- default value: `Referer` is sent always, unless we send a request from HTTPS to HTTP (to less secure protocol). -- **`"no-referrer"`** -- never send `Referer`. -- **`"origin"`** -- only send the origin in `Referer`, not the full page URL, e.g. `http://site.com` instead of `http://site.com/path`. -- **`"origin-when-cross-origin"`** -- send full referrer to the same origin, but only the origin part for cross-origin requests. -- **`"same-origin"`** -- send full referrer to the same origin, but no referer for for cross-origin requests. -- **`"strict-origin"`** -- send only origin, don't send referrer for HTTPS→HTTP requests. -- **`"strict-origin-when-cross-origin"`** -- for same-origin send full referrer, for cross-origin send only origin, unless it's HTTPS→HTTP request, then send nothing. -- **`"unsafe-url"`** -- always send full url in `Referer`. - -Let's say we have an admin zone with URL structure that shouldn't be visible from outside. - -If we send a cross-origin `fetch`, then by default it sends the `Referer` header with the full url of our page (except when we request from HTTPS to HTTP, then no `Referer`). - -E.g. `Referer: https://javascript.info/admin/secret/paths`. - -If we'd like to totally hide the referrer: - -```js -fetch('https://another.com/page', { - referrerPolicy: "no-referrer" // no Referer, same effect as referrer: "" -}); -``` - -Otherwise, if we'd like the remote side to see where the request comes from, we can send only the "origin" part of the url: - -```js -fetch('https://another.com/page', { - referrerPolicy: "strict-origin" // Referer: https://javascript.info -}); -``` - -## mode - -The `mode` option serves as a safe-guard that prevents cross-origin requests: - -- **`"cors"`** -- the default, cross-origin requests are allowed, as described in <info:fetch-crossorigin>, -- **`"same-origin"`** -- cross-origin requests are forbidden, -- **`"no-cors"`** -- only simple cross-origin requests are allowed. - -That may be useful in contexts when the fetch url comes from 3rd-party, and we want a "power off switch" to limit cross-origin capabilities. - -## credentials - -The `credentials` option specifies whether `fetch` should send cookies and HTTP-Authorization headers with the request. - -- **`"same-origin"`** -- the default, don't send for cross-origin requests, -- **`"include"`** -- always send, requires `Accept-Control-Allow-Credentials` from cross-origin server, -- **`"omit"`** -- never send, even for same-origin requests. - -## cache - -By default, `fetch` requests make use of standard HTTP-caching. That is, it honors `Expires`, `Cache-Control` headers, sends `If-Modified-Since`, and so on. Just like regular HTTP-requests do. - -The `cache` options allows to ignore HTTP-cache or fine-tune its usage: - -- **`"default"`** -- `fetch` uses standard HTTP-cache rules and headers; -- **`"no-store"`** -- totally ignore HTTP-cache, this mode becomes the default if we set a header `If-Modified-Since`, `If-None-Match`, `If-Unmodified-Since`, `If-Match`, or `If-Range`; -- **`"reload"`** -- don't take the result from HTTP-cache (if any), but populate cache with the response (if response headers allow); -- **`"no-cache"`** -- create a conditional request if there is a cached response, and a normal request otherwise. Populate HTTP-cache with the response; -- **`"force-cache"`** -- use a response from HTTP-cache, even if it's stale. If there's no response in HTTP-cache, make a regular HTTP-request, behave normally; -- **`"only-if-cached"`** -- use a response from HTTP-cache, even if it's stale. If there's no response in HTTP-cache, then error. Only works when `mode` is `"same-origin"`. - -## redirect - -Normally, `fetch` transparently follows HTTP-redirects, like 301, 302 etc. - -The `redirect` option allows to change that: - -- **`"follow"`** -- the default, follow HTTP-redirects, -- **`"error"`** -- error in case of HTTP-redirect, -- **`"manual"`** -- don't follow HTTP-redirect, but `response.url` will be the new URL, and `response.redirected` will be `true`, so that we can perform the redirect manually to the new URL (if needed). - -## integrity - -The `integrity` option allows to check if the response matches the known-ahead checksum. - -As described in the [specification](https://w3c.github.io/webappsec-subresource-integrity/), supported hash-functions are SHA-256, SHA-384, and SHA-512, there might be others depending on a browser. - -For example, we're downloading a file, and we know that it's SHA-256 checksum is "abc" (a real checksum is longer, of course). - -We can put it in the `integrity` option, like this: - -```js -fetch('http://site.com/file', { - integrity: 'sha256-abd' -}); -``` - -Then `fetch` will calculate SHA-256 on its own and compare it with our string. In case of a mismatch, an error is triggered. - -## keepalive - -The `keepalive` option indicates that the request may outlive the page. - -For example, we gather statistics about how the current visitor uses our page (mouse clicks, page fragments he views), to improve user experience. - -When the visitor leaves our page -- we'd like to save it on our server. - -We can use `window.onunload` for that: - -```js run -window.onunload = function() { - fetch('/analytics', { - method: 'POST', - body: "statistics", -*!* - keepalive: true -*/!* - }); -}; -``` - -Normally, when a document is unloaded, all associated network requests are aborted. But `keepalive` option tells the browser to perform the request in background, even after it leaves the page. So it's essential for our request to succeed. - -- We can't send megabytes: the body limit for keepalive requests is 64kb. - - If we gather more data, we can send it out regularly, then there won't be a lot for the "onunload" request. - - The limit is for all currently ongoing requests. So we cheat it by creating 100 requests, each 64kb. -- We don't get the server response if the request is made `onunload`, because the document is already unloaded at that time. - - Usually, the server sends empty response to such requests, so it's not a problem. diff --git a/5-network/04-fetch-crossorigin/1-do-we-need-origin/solution.md b/5-network/05-fetch-crossorigin/1-do-we-need-origin/solution.md similarity index 79% rename from 5-network/04-fetch-crossorigin/1-do-we-need-origin/solution.md rename to 5-network/05-fetch-crossorigin/1-do-we-need-origin/solution.md index 8cbb2c88a..d44494e18 100644 --- a/5-network/04-fetch-crossorigin/1-do-we-need-origin/solution.md +++ b/5-network/05-fetch-crossorigin/1-do-we-need-origin/solution.md @@ -2,7 +2,7 @@ We need `Origin`, because sometimes `Referer` is absent. For instance, when we ` The [Content Security Policy](http://en.wikipedia.org/wiki/Content_Security_Policy) may forbid sending a `Referer`. -As we'll see, `fetch` also has options that prevent sending the `Referer` and even allow to change it (within the same site). +As we'll see, `fetch` has options that prevent sending the `Referer` and even allow to change it (within the same site). By specification, `Referer` is an optional HTTP-header. diff --git a/5-network/04-fetch-crossorigin/1-do-we-need-origin/task.md b/5-network/05-fetch-crossorigin/1-do-we-need-origin/task.md similarity index 89% rename from 5-network/04-fetch-crossorigin/1-do-we-need-origin/task.md rename to 5-network/05-fetch-crossorigin/1-do-we-need-origin/task.md index 6bfdc1094..fb8142a3b 100644 --- a/5-network/04-fetch-crossorigin/1-do-we-need-origin/task.md +++ b/5-network/05-fetch-crossorigin/1-do-we-need-origin/task.md @@ -25,4 +25,4 @@ As you can see, both `Referer` and `Origin` are present. The questions: 1. Why `Origin` is needed, if `Referer` has even more information? -2. If it possible that there's no `Referer` or `Origin`, or it's incorrect? +2. Is it possible that there's no `Referer` or `Origin`, or is it incorrect? diff --git a/5-network/05-fetch-crossorigin/article.md b/5-network/05-fetch-crossorigin/article.md new file mode 100644 index 000000000..4420f43c7 --- /dev/null +++ b/5-network/05-fetch-crossorigin/article.md @@ -0,0 +1,384 @@ +# Fetch: Cross-Origin Requests + +If we send a `fetch` request to another web-site, it will probably fail. + +For instance, let's try fetching `http://example.com`: + +```js run async +try { + await fetch('http://example.com'); +} catch(err) { + alert(err); // Failed to fetch +} +``` + +Fetch fails, as expected. + +The core concept here is *origin* -- a domain/port/protocol triplet. + +Cross-origin requests -- those sent to another domain (even a subdomain) or protocol or port -- require special headers from the remote side. + +That policy is called "CORS": Cross-Origin Resource Sharing. + +## Why is CORS needed? A brief history + +CORS exists to protect the internet from evil hackers. + +Seriously. Let's make a very brief historical digression. + +**For many years a script from one site could not access the content of another site.** + +That simple, yet powerful rule was a foundation of the internet security. E.g. an evil script from website `hacker.com` could not access the user's mailbox at website `gmail.com`. People felt safe. + +JavaScript also did not have any special methods to perform network requests at that time. It was a toy language to decorate a web page. + +But web developers demanded more power. A variety of tricks were invented to work around the limitation and make requests to other websites. + +### Using forms + +One way to communicate with another server was to submit a `<form>` there. People submitted it into `<iframe>`, just to stay on the current page, like this: + +```html +<!-- form target --> +*!* +<iframe name="iframe"></iframe> +*/!* + +<!-- a form could be dynamically generated and submitted by JavaScript --> +*!* +<form target="iframe" method="POST" action="http://another.com/…"> +*/!* + ... +</form> +``` + +So, it was possible to make a GET/POST request to another site, even without networking methods, as forms can send data anywhere. But as it's forbidden to access the content of an `<iframe>` from another site, it wasn't possible to read the response. + +To be precise, there were actually tricks for that, they required special scripts at both the iframe and the page. So the communication with the iframe was technically possible. Right now there's no point to go into details, let these dinosaurs rest in peace. + +### Using scripts + +Another trick was to use a `script` tag. A script could have any `src`, with any domain, like `<script src="http://another.com/…">`. It's possible to execute a script from any website. + +If a website, e.g. `another.com` intended to expose data for this kind of access, then a so-called "JSONP (JSON with padding)" protocol was used. + +Here's how it worked. + +Let's say we, at our site, need to get the data from `http://another.com`, such as the weather: + +1. First, in advance, we declare a global function to accept the data, e.g. `gotWeather`. + + ```js + // 1. Declare the function to process the weather data + function gotWeather({ temperature, humidity }) { + alert(`temperature: ${temperature}, humidity: ${humidity}`); + } + ``` +2. Then we make a `<script>` tag with `src="http://another.com/weather.json?callback=gotWeather"`, using the name of our function as the `callback` URL-parameter. + + ```js + let script = document.createElement('script'); + script.src = `http://another.com/weather.json?callback=gotWeather`; + document.body.append(script); + ``` +3. The remote server `another.com` dynamically generates a script that calls `gotWeather(...)` with the data it wants us to receive. + ```js + // The expected answer from the server looks like this: + gotWeather({ + temperature: 25, + humidity: 78 + }); + ``` +4. When the remote script loads and executes, `gotWeather` runs, and, as it's our function, we have the data. + +That works, and doesn't violate security, because both sides agreed to pass the data this way. And, when both sides agree, it's definitely not a hack. There are still services that provide such access, as it works even for very old browsers. + +After a while, networking methods appeared in browser JavaScript. + +At first, cross-origin requests were forbidden. But as a result of long discussions, cross-origin requests were allowed, but with any new capabilities requiring an explicit allowance by the server, expressed in special headers. + +## Safe requests + +There are two types of cross-origin requests: + +1. Safe requests. +2. All the others. + +Safe Requests are simpler to make, so let's start with them. + +A request is safe if it satisfies two conditions: + +1. [Safe method](https://fetch.spec.whatwg.org/#cors-safelisted-method): GET, POST or HEAD +2. [Safe headers](https://fetch.spec.whatwg.org/#cors-safelisted-request-header) -- the only allowed custom headers are: + - `Accept`, + - `Accept-Language`, + - `Content-Language`, + - `Content-Type` with the value `application/x-www-form-urlencoded`, `multipart/form-data` or `text/plain`. + +Any other request is considered "unsafe". For instance, a request with `PUT` method or with an `API-Key` HTTP-header does not fit the limitations. + +**The essential difference is that a safe request can be made with a `<form>` or a `<script>`, without any special methods.** + +So, even a very old server should be ready to accept a safe request. + +Contrary to that, requests with non-standard headers or e.g. method `DELETE` can't be created this way. For a long time JavaScript was unable to do such requests. So an old server may assume that such requests come from a privileged source, "because a webpage is unable to send them". + +When we try to make a unsafe request, the browser sends a special "preflight" request that asks the server -- does it agree to accept such cross-origin requests, or not? + +And, unless the server explicitly confirms that with headers, an unsafe request is not sent. + +Now we'll go into details. + +## CORS for safe requests + +If a request is cross-origin, the browser always adds the `Origin` header to it. + +For instance, if we request `https://anywhere.com/request` from `https://javascript.info/page`, the headers will look like: + +```http +GET /request +Host: anywhere.com +*!* +Origin: https://javascript.info +*/!* +... +``` + +As you can see, the `Origin` header contains exactly the origin (domain/protocol/port), without a path. + +The server can inspect the `Origin` and, if it agrees to accept such a request, add a special header `Access-Control-Allow-Origin` to the response. That header should contain the allowed origin (in our case `https://javascript.info`), or a star `*`. Then the response is successful, otherwise it's an error. + +The browser plays the role of a trusted mediator here: +1. It ensures that the correct `Origin` is sent with a cross-origin request. +2. It checks for permitting `Access-Control-Allow-Origin` in the response, if it exists, then JavaScript is allowed to access the response, otherwise it fails with an error. + + + +Here's an example of a permissive server response: +```http +200 OK +Content-Type:text/html; charset=UTF-8 +*!* +Access-Control-Allow-Origin: https://javascript.info +*/!* +``` + +## Response headers + +For cross-origin request, by default JavaScript may only access so-called "safe" response headers: + +- `Cache-Control` +- `Content-Language` +- `Content-Length` +- `Content-Type` +- `Expires` +- `Last-Modified` +- `Pragma` + +Accessing any other response header causes an error. + +To grant JavaScript access to any other response header, the server must send the `Access-Control-Expose-Headers` header. It contains a comma-separated list of unsafe header names that should be made accessible. + +For example: + +```http +200 OK +Content-Type:text/html; charset=UTF-8 +Content-Length: 12345 +Content-Encoding: gzip +API-Key: 2c9de507f2c54aa1 +Access-Control-Allow-Origin: https://javascript.info +*!* +Access-Control-Expose-Headers: Content-Encoding,API-Key +*/!* +``` + +With such an `Access-Control-Expose-Headers` header, the script is allowed to read the `Content-Encoding` and `API-Key` headers of the response. + +## "Unsafe" requests + +We can use any HTTP-method: not just `GET/POST`, but also `PATCH`, `DELETE` and others. + +Some time ago no one could even imagine that a webpage could make such requests. So there may still exist webservices that treat a non-standard method as a signal: "That's not a browser". They can take it into account when checking access rights. + +So, to avoid misunderstandings, any "unsafe" request -- that couldn't be done in the old times, the browser does not make such requests right away. First, it sends a preliminary, so-called "preflight" request, to ask for permission. + +A preflight request uses the method `OPTIONS`, no body and three headers: + +- `Access-Control-Request-Method` header has the method of the unsafe request. +- `Access-Control-Request-Headers` header provides a comma-separated list of its unsafe HTTP-headers. +- `Origin` header tells from where the request came. (such as `https://javascript.info`) + +If the server agrees to serve the requests, then it should respond with empty body, status 200 and headers: + +- `Access-Control-Allow-Origin` must be either `*` or the requesting origin, such as `https://javascript.info`, to allow it. +- `Access-Control-Allow-Methods` must have the allowed method. +- `Access-Control-Allow-Headers` must have a list of allowed headers. +- Additionally, the header `Access-Control-Max-Age` may specify a number of seconds to cache the permissions. So the browser won't have to send a preflight for subsequent requests that satisfy given permissions. + + + +Let's see how it works step-by-step on the example of a cross-origin `PATCH` request (this method is often used to update data): + +```js +let response = await fetch('https://site.com/service.json', { + method: 'PATCH', + headers: { + 'Content-Type': 'application/json', + 'API-Key': 'secret' + } +}); +``` + +There are three reasons why the request is unsafe (one is enough): +- Method `PATCH` +- `Content-Type` is not one of: `application/x-www-form-urlencoded`, `multipart/form-data`, `text/plain`. +- "Unsafe" `API-Key` header. + +### Step 1 (preflight request) + +Prior to sending such a request, the browser, on its own, sends a preflight request that looks like this: + +```http +OPTIONS /service.json +Host: site.com +Origin: https://javascript.info +Access-Control-Request-Method: PATCH +Access-Control-Request-Headers: Content-Type,API-Key +``` + +- Method: `OPTIONS`. +- The path -- exactly the same as the main request: `/service.json`. +- Cross-origin special headers: + - `Origin` -- the source origin. + - `Access-Control-Request-Method` -- requested method. + - `Access-Control-Request-Headers` -- a comma-separated list of "unsafe" headers. + +### Step 2 (preflight response) + +The server should respond with status 200 and the headers: +- `Access-Control-Allow-Origin: https://javascript.info` +- `Access-Control-Allow-Methods: PATCH` +- `Access-Control-Allow-Headers: Content-Type,API-Key`. + +That allows future communication, otherwise an error is triggered. + +If the server expects other methods and headers in the future, it makes sense to allow them in advance by adding them to the list. + +For example, this response also allows `PUT`, `DELETE` and additional headers: + +```http +200 OK +Access-Control-Allow-Origin: https://javascript.info +Access-Control-Allow-Methods: PUT,PATCH,DELETE +Access-Control-Allow-Headers: API-Key,Content-Type,If-Modified-Since,Cache-Control +Access-Control-Max-Age: 86400 +``` + +Now the browser can see that `PATCH` is in `Access-Control-Allow-Methods` and `Content-Type,API-Key` are in the list `Access-Control-Allow-Headers`, so it sends out the main request. + +If there's the header `Access-Control-Max-Age` with a number of seconds, then the preflight permissions are cached for the given time. The response above will be cached for 86400 seconds (one day). Within this timeframe, subsequent requests will not cause a preflight. Assuming that they fit the cached allowances, they will be sent directly. + +### Step 3 (actual request) + +When the preflight is successful, the browser now makes the main request. The process here is the same as for safe requests. + +The main request has the `Origin` header (because it's cross-origin): + +```http +PATCH /service.json +Host: site.com +Content-Type: application/json +API-Key: secret +Origin: https://javascript.info +``` + +### Step 4 (actual response) + +The server should not forget to add `Access-Control-Allow-Origin` to the main response. A successful preflight does not relieve from that: + +```http +Access-Control-Allow-Origin: https://javascript.info +``` + +Then JavaScript is able to read the main server response. + +```smart +Preflight request occurs "behind the scenes", it's invisible to JavaScript. + +JavaScript only gets the response to the main request or an error if there's no server permission. +``` + +## Credentials + +A cross-origin request initiated by JavaScript code by default does not bring any credentials (cookies or HTTP authentication). + +That's uncommon for HTTP-requests. Usually, a request to `http://site.com` is accompanied by all cookies from that domain. Cross-origin requests made by JavaScript methods on the other hand are an exception. + +For example, `fetch('http://another.com')` does not send any cookies, even those (!) that belong to `another.com` domain. + +Why? + +That's because a request with credentials is much more powerful than without them. If allowed, it grants JavaScript the full power to act on behalf of the user and access sensitive information using their credentials. + +Does the server really trust the script that much? Then it must explicitly allow requests with credentials with an additional header. + +To send credentials in `fetch`, we need to add the option `credentials: "include"`, like this: + +```js +fetch('http://another.com', { + credentials: "include" +}); +``` + +Now `fetch` sends cookies originating from `another.com` with request to that site. + +If the server agrees to accept the request *with credentials*, it should add a header `Access-Control-Allow-Credentials: true` to the response, in addition to `Access-Control-Allow-Origin`. + +For example: + +```http +200 OK +Access-Control-Allow-Origin: https://javascript.info +Access-Control-Allow-Credentials: true +``` + +Please note: `Access-Control-Allow-Origin` is prohibited from using a star `*` for requests with credentials. Like shown above, it must provide the exact origin there. That's an additional safety measure, to ensure that the server really knows who it trusts to make such requests. + +## Summary + +From the browser point of view, there are two kinds of cross-origin requests: "safe" and all the others. + +"Safe" requests must satisfy the following conditions: +- Method: GET, POST or HEAD. +- Headers -- we can set only: + - `Accept` + - `Accept-Language` + - `Content-Language` + - `Content-Type` to the value `application/x-www-form-urlencoded`, `multipart/form-data` or `text/plain`. + +The essential difference is that safe requests were doable since ancient times using `<form>` or `<script>` tags, while unsafe were impossible for browsers for a long time. + +So, the practical difference is that safe requests are sent right away, with the `Origin` header, while for the other ones the browser makes a preliminary "preflight" request, asking for permission. + +**For safe requests:** + +- → The browser sends the `Origin` header with the origin. +- ← For requests without credentials (not sent by default), the server should set: + - `Access-Control-Allow-Origin` to `*` or same value as `Origin` +- ← For requests with credentials, the server should set: + - `Access-Control-Allow-Origin` to same value as `Origin` + - `Access-Control-Allow-Credentials` to `true` + +Additionally, to grant JavaScript access to any response headers except `Cache-Control`, `Content-Language`, `Content-Type`, `Expires`, `Last-Modified` or `Pragma`, the server should list the allowed ones in `Access-Control-Expose-Headers` header. + +**For unsafe requests, a preliminary "preflight" request is issued before the requested one:** + +- → The browser sends an `OPTIONS` request to the same URL, with the headers: + - `Access-Control-Request-Method` has requested method. + - `Access-Control-Request-Headers` lists unsafe requested headers. +- ← The server should respond with status 200 and the headers: + - `Access-Control-Allow-Methods` with a list of allowed methods, + - `Access-Control-Allow-Headers` with a list of allowed headers, + - `Access-Control-Max-Age` with a number of seconds to cache the permissions. +- Then the actual request is sent, and the previous "safe" scheme is applied. diff --git a/5-network/04-fetch-crossorigin/cors-gmail-messages.svg b/5-network/05-fetch-crossorigin/cors-gmail-messages.svg similarity index 73% rename from 5-network/04-fetch-crossorigin/cors-gmail-messages.svg rename to 5-network/05-fetch-crossorigin/cors-gmail-messages.svg index cfe15d7ad..5779836d0 100644 --- a/5-network/04-fetch-crossorigin/cors-gmail-messages.svg +++ b/5-network/05-fetch-crossorigin/cors-gmail-messages.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:5-network/04-fetch-crossorigin/cors-gmail-messages.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="651px" height="285px" viewBox="0 0 651 285" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -51,4 +52,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="651" height="285" viewBox="0 0 651 285"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="cors-gmail-messages.svg"><path id="Rectangle-1" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M11 48h298v199H11z"/><g id="<script>-let-respons" fill-rule="nonzero" transform="translate(15.937 58.259)"><path id="<script>" fill="#7E7C7B" d="M5.339 9.099l-.745.752L0 6.016 4.594 2.18l.745.758-3.726 3.063L5.34 9.099zm7.95-1.23a1.634 1.634 0 01-.458 1.158 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199V8.511c.4.114.8.2 1.196.26.397.059.79.088 1.183.088.57 0 .991-.077 1.264-.232.274-.155.41-.376.41-.663a.812.812 0 00-.064-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.372.54.075.2.113.428.113.683zm7.595 1.62c-.31.118-.628.205-.954.262a5.845 5.845 0 01-1.008.086c-1.085 0-1.92-.294-2.505-.882-.586-.588-.879-1.447-.879-2.577 0-.542.084-1.035.253-1.477a3.142 3.142 0 011.805-1.863c.424-.17.89-.256 1.401-.256.356 0 .688.025.998.075.31.05.606.133.889.247v1.134a3.77 3.77 0 00-.906-.338 4.248 4.248 0 00-.954-.106c-.305 0-.593.058-.864.174-.271.116-.51.284-.715.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.202 1.45.605 1.863.404.412.963.618 1.678.618a3.897 3.897 0 001.818-.438v1.108zm2.645-6.611h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.617.69.367.46.536 1.144.509 2.051h-1.203c.013-.601-.074-1.038-.263-1.309-.19-.271-.466-.407-.83-.407-.16 0-.321.029-.483.086a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.616.711v4.41H23.53V2.877zm9.755.984h-2.03v-.984h3.233V8.75h2.044v.991h-5.503V8.75h2.256V3.862zM33.701 0c.132 0 .255.024.37.072a.893.893 0 01.297.201.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.298.202.944.944 0 01-.369.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.664.893.893 0 01.297-.201.944.944 0 01.37-.072zM44.51 6.187c0 .61-.086 1.143-.257 1.6-.17.455-.406.833-.707 1.134a2.9 2.9 0 01-1.066.677c-.41.15-.855.225-1.333.225-.22 0-.437-.011-.653-.034a4.952 4.952 0 01-.66-.116v2.87h-1.19V2.879h1.06l.075 1.148c.342-.469.707-.798 1.094-.987.387-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.545.395.745.694.2.298.351.658.451 1.08.1.421.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.991a2.586 2.586 0 00-.25-.772 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.635-.181c-.15 0-.303.024-.458.072a1.857 1.857 0 00-.482.239 3.59 3.59 0 00-.527.444 7 7 0 00-.591.687v3.33c.219.09.449.162.69.215.242.052.479.078.711.078.643 0 1.147-.217 1.511-.652.365-.436.547-1.09.547-1.966zm8.58 3.405a5.793 5.793 0 01-.835.146 8.52 8.52 0 01-.875.045c-.861 0-1.504-.195-1.928-.585-.423-.39-.635-.987-.635-1.794V3.876h-1.921v-.998h1.92V.99l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.131.86.393 1.104.262.244.648.366 1.159.366.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025zm2.501-6.707l.738-.758 4.594 3.835-4.594 3.835-.738-.752 3.726-3.07-3.726-3.09z"/><path id="let" fill="#1C85B5" d="M2.495 33.06H.465v-.978h3.233v8.668h2.044v.991H.24v-.991h2.256v-7.69zm11.156 4.887c0 .169-.002.31-.006.424a6.37 6.37 0 01-.021.321h-4.82c0 .702.197 1.241.589 1.617.392.376.957.564 1.695.564.2 0 .4-.008.601-.024.201-.016.395-.037.582-.065.186-.027.365-.058.536-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.069-.94.208-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.575 1.005.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.71.15-.215.1-.4.242-.555.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.177.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.861 0-1.504-.195-1.928-.585-.424-.39-.635-.987-.635-1.794v-3.582h-1.921v-.998h1.92V32.99l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.131.86.393 1.104.262.244.648.366 1.159.366.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025z"/><path id="response" fill="#181717" d="M31.227 34.878h1.086l.035 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051h-1.204c.014-.601-.074-1.038-.263-1.309-.19-.271-.466-.407-.83-.407-.16 0-.32.029-.483.086a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.213 3.07c0 .168-.002.31-.006.423a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.585.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.575 1.005.13.394.194.835.194 1.322zm-1.237-.172a2.593 2.593 0 00-.089-.837 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.1-.4.242-.555.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.177.84h3.61zm8.572 2.092a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.197.26.396.059.79.088 1.182.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.45-.56 1.649 1.649 0 01-.165-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.367-.195.27 0 .568.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.373.54.075.2.112.428.112.683zm8.128-1.681c0 .61-.085 1.143-.256 1.6-.17.455-.407.833-.708 1.134a2.9 2.9 0 01-1.066.677c-.41.15-.854.225-1.333.225-.219 0-.436-.011-.653-.034a4.952 4.952 0 01-.66-.116v2.87h-1.189v-9.665h1.06l.075 1.148c.342-.469.706-.798 1.094-.987.387-.19.806-.284 1.257-.284.392 0 .736.082 1.033.246.296.164.544.395.745.694.2.298.35.658.45 1.08.101.421.151.892.151 1.412zm-1.216.054c0-.36-.027-.69-.08-.991a2.586 2.586 0 00-.249-.772 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.636-.181c-.15 0-.303.024-.458.072a1.857 1.857 0 00-.482.239 3.59 3.59 0 00-.526.444 7 7 0 00-.591.687v3.33c.218.09.448.162.69.215.242.052.478.078.71.078.643 0 1.147-.217 1.512-.652.364-.436.547-1.09.547-1.966zm9.023.014a4.55 4.55 0 01-.226 1.466c-.15.445-.366.825-.649 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.381.263c-.492 0-.933-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.622-1.1c-.144-.438-.215-.942-.215-1.511 0-.534.075-1.02.225-1.46.15-.44.367-.818.65-1.135a2.913 2.913 0 011.032-.738 3.447 3.447 0 011.38-.263c.493 0 .934.076 1.323.229.39.153.72.376.992.67.27.294.478.66.622 1.097.143.437.215.939.215 1.504zm-1.217.055c0-.424-.047-.795-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.632-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.356.81 4.06 4.06 0 00-.113.968c0 .423.047.795.14 1.114.094.319.227.584.4.796.174.212.383.372.63.479.245.107.523.16.833.16.356 0 .66-.07.913-.208.253-.14.46-.325.622-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm2.94-3.432h1.06l.047 1.107c.2-.237.394-.434.581-.591.187-.157.37-.284.55-.38a2.06 2.06 0 01.55-.201c.188-.039.381-.058.582-.058.706 0 1.24.208 1.603.625.362.417.543 1.045.543 1.884v4.477h-1.19V37.36c0-.537-.1-.935-.3-1.192-.2-.258-.499-.387-.896-.387-.145 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.313.252-.489.434-.175.182-.372.41-.59.683v4.553h-1.19v-6.863zm13.131 4.99a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.197.26.396.059.79.088 1.182.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.367-.195.27 0 .568.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.406.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.373.54.075.2.112.428.112.683zm8.06-1.92c0 .168-.002.31-.007.423a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.696.564.2 0 .4-.008.601-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.225.37.15.681.364.934.64.253.275.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.172a2.593 2.593 0 00-.09-.837 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.1-.399.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61z"/><path id="=" fill="#DBAF88" d="M105.916 37.325h-5.879V36.32h5.879v1.005zm0 2.38h-5.879v-1.006h5.879v1.005z"/><path id="await" fill="#1C85B5" d="M120.025 41.741l-.027-.923c-.374.37-.753.636-1.138.8-.385.164-.79.246-1.214.246-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.397-.618 2.289 2.289 0 01-.126-.773c0-.688.256-1.227.769-1.617.512-.39 1.27-.584 2.273-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.642-.1c.224-.025.45-.037.677-.037.415 0 .789.045 1.121.137.333.09.614.23.844.417.23.186.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.058-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .132.021.259.062.38.041.12.107.227.198.32.091.094.21.168.356.223.146.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm9.776-3.74l-.998 6.864h-1.443l-.99-2.87-.199-.698-.226.738-.95 2.83h-1.401l-.991-6.863h1.162l.574 4.662.123 1.04.294-.91.998-3.083h.855l1.073 3.042.307.91.103-.965.533-4.696h1.176zm5.749 6.864l-.027-.923c-.374.37-.754.636-1.139.8-.385.164-.79.246-1.213.246-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.396-.618 2.289 2.289 0 01-.127-.773c0-.688.257-1.227.77-1.617.512-.39 1.27-.584 2.272-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.643-.1c.223-.025.449-.037.677-.037.414 0 .788.045 1.12.137.333.09.615.23.845.417.23.186.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.058-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .132.02.259.062.38.04.12.107.227.198.32.09.094.21.168.355.223.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm5.756-2.755h-2.03v-.984h3.233v5.872h2.044v.991h-5.503v-.991h2.256v-4.888zm.417-3.862c.132 0 .255.024.369.072a.893.893 0 01.297.201.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.202.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.664.893.893 0 01.298-.201.944.944 0 01.369-.072zm10.473 9.646a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.862 0-1.504-.195-1.928-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.92v-.998h1.92V32.99l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.13.86.393 1.104.262.244.648.366 1.158.366.219 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025z"/><path id="fetch" fill="#181717" d="M167.89 33.19c-.624-.133-1.161-.199-1.613-.199-1.07 0-1.606.56-1.606 1.682v1.203h3.008v.991h-3.008v4.874h-1.21v-4.874h-2.208v-.991h2.208v-1.135c0-1.827.952-2.741 2.857-2.741.474 0 .998.055 1.573.164v1.025zm7.404 4.757c0 .169-.002.31-.007.424a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.201 0 .402-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.225.37.15.681.364.934.64.253.275.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.1-.4.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.862 0-1.504-.195-1.928-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.92v-.998h1.92V32.99l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.13.86.393 1.104.262.244.648.366 1.158.366.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm7.499-.158c-.31.119-.628.206-.954.263a5.845 5.845 0 01-1.008.086c-1.085 0-1.92-.294-2.506-.882-.585-.588-.878-1.447-.878-2.577 0-.542.084-1.035.253-1.477a3.142 3.142 0 011.805-1.863c.423-.17.89-.256 1.4-.256.356 0 .69.025.999.075.31.05.606.133.889.247v1.134a3.77 3.77 0 00-.906-.338 4.248 4.248 0 00-.954-.106c-.305 0-.593.058-.864.174-.272.116-.51.284-.715.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.202 1.45.605 1.863.403.412.963.618 1.678.618a3.897 3.897 0 001.818-.438v1.108zm7.881.253h-1.189V37.36c0-.528-.1-.924-.297-1.186-.199-.262-.482-.393-.851-.393-.16 0-.31.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.616.683v4.553h-1.19v-9.659h1.19v2.796l-.04 1.08c.186-.223.37-.411.55-.564.18-.153.36-.277.54-.373.18-.095.363-.164.55-.205.187-.04.38-.061.581-.061.684 0 1.212.208 1.586.625.374.417.56 1.045.56 1.884v4.477z"/><path id="(" fill="#7E7C7B" d="M204.395 44.612c-2.101-1.946-3.152-4.097-3.152-6.453 0-.551.056-1.102.168-1.65.111-.55.291-1.1.54-1.652a9.028 9.028 0 01.984-1.654c.408-.551.904-1.098 1.487-1.64l.69.704c-1.768 1.745-2.652 3.68-2.652 5.803 0 1.058.223 2.074.67 3.05.447.974 1.107 1.895 1.982 2.76l-.717.732z"/><path id="'https://gmail.com/messages'" fill="#478964" d="M19.093 46.082l-.198 3.206h-1.176l-.192-3.206h1.566zm9.673 9.66h-1.19v-4.383c0-.528-.099-.924-.297-1.186-.198-.262-.482-.393-.851-.393-.16 0-.309.022-.448.065a1.557 1.557 0 00-.434.226 3.73 3.73 0 00-.492.434c-.178.182-.383.41-.616.683v4.553H23.25v-9.659h1.19v2.796l-.042 1.08c.187-.223.37-.411.55-.564.18-.153.36-.277.54-.373.18-.095.364-.164.551-.205.187-.04.38-.061.581-.061.684 0 1.212.208 1.586.625.374.417.56 1.045.56 1.884v4.477zm7.71-.096a5.793 5.793 0 01-.833.146 8.52 8.52 0 01-.875.045c-.862 0-1.504-.195-1.928-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.92v-.998h1.92V46.99l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.13.86.393 1.104.262.244.648.366 1.158.366.22 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025zm7.698 0a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.861 0-1.504-.195-1.928-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.92v-.998h1.92V46.99l1.19-.307v2.194h3.083v.998H41.09v3.486c0 .492.13.86.393 1.104.262.244.648.366 1.159.366.218 0 .458-.017.717-.051.26-.034.531-.088.814-.16v1.025zm8.032-3.46c0 .611-.085 1.144-.256 1.6-.171.456-.407.834-.708 1.135a2.9 2.9 0 01-1.066.677c-.41.15-.855.225-1.333.225-.219 0-.437-.011-.653-.034a4.952 4.952 0 01-.66-.116v2.87h-1.19v-9.665h1.06l.076 1.148c.341-.469.706-.798 1.093-.987.388-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.297.164.545.395.745.694.201.298.351.658.452 1.08.1.421.15.892.15 1.412zm-1.217.055c0-.36-.026-.69-.078-.991a2.586 2.586 0 00-.25-.772 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.636-.181c-.15 0-.303.024-.458.072a1.857 1.857 0 00-.482.239 3.59 3.59 0 00-.526.444 7 7 0 00-.592.687v3.33c.219.09.45.162.69.215.242.052.48.078.712.078.642 0 1.146-.217 1.51-.652.365-.436.547-1.09.547-1.966zm8.484 1.627a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.493.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.402.114.8.2 1.197.26.396.059.79.088 1.182.088.57 0 .992-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.368-.195.268 0 .567.015.895.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.232.094.535.196.91.305.405.118.747.243 1.025.372.278.13.503.275.676.435.173.159.298.339.373.54.075.2.113.428.113.683zm5.017-5.113c.142 0 .276.028.404.085a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.308c-.146 0-.281-.028-.407-.083a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.23.971.971 0 01.407-.085zm0 5.01c.142 0 .276.03.404.086a1.107 1.107 0 01.564.564.978.978 0 01.085.403.971.971 0 01-.085.407 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.307c-.146 0-.281-.027-.407-.082a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.407 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.331-.229.971.971 0 01.407-.085zm10.46-7.683l-4.69 11.129h-1.135l4.69-11.129h1.134zm7.696 0l-4.689 11.129h-1.135l4.69-11.129h1.134zm7.124 3.767c.127.16.226.345.297.557.07.212.106.44.106.687 0 .355-.065.68-.195.974-.13.294-.313.546-.55.755a2.54 2.54 0 01-.851.489c-.33.116-.694.174-1.09.174-.288 0-.557-.03-.807-.092a2.064 2.064 0 01-.595-.229c-.087.128-.16.248-.219.362a.817.817 0 00-.089.383c0 .173.084.317.25.43.166.115.386.176.66.185l1.804.069c.342.009.658.052.947.13.29.077.538.189.745.335.207.145.37.325.485.54.117.214.175.462.175.745 0 .305-.066.594-.198.868a1.985 1.985 0 01-.612.721c-.276.207-.626.373-1.05.496-.423.123-.927.184-1.51.184-.556 0-1.03-.044-1.419-.133-.39-.089-.71-.212-.96-.37a1.49 1.49 0 01-.547-.56 1.51 1.51 0 01-.171-.714c0-.333.077-.624.232-.875.155-.25.395-.492.718-.725a1.133 1.133 0 01-.52-.475 1.322 1.322 0 01-.164-.632c0-.296.07-.567.21-.814.138-.246.303-.478.495-.697a3.57 3.57 0 01-.23-.307 1.91 1.91 0 01-.28-.701 2.536 2.536 0 01-.037-.462c0-.355.065-.68.195-.974.13-.294.312-.545.547-.755.234-.21.517-.373.847-.489.33-.116.696-.174 1.097-.174.169 0 .33.011.486.034.155.023.291.052.41.089h2.488v.97h-1.1zm-4.13 6.883c0 .324.17.56.506.708.338.148.807.222 1.409.222.378 0 .696-.034.953-.102.258-.069.465-.159.622-.27.158-.112.27-.24.339-.383.068-.144.102-.29.102-.441 0-.278-.114-.483-.342-.615-.227-.133-.576-.21-1.045-.233l-1.791-.061c-.15.1-.275.198-.373.294a1.292 1.292 0 00-.229.29 1.127 1.127 0 00-.116.294 1.309 1.309 0 00-.034.297zm.363-5.612c0 .219.036.42.11.602.072.182.175.337.307.465.132.127.288.226.468.297.18.07.38.106.598.106.237 0 .448-.04.633-.12.184-.08.34-.188.464-.324.126-.137.222-.294.288-.472.066-.178.099-.362.099-.554 0-.219-.037-.42-.11-.601a1.316 1.316 0 00-.307-.465 1.404 1.404 0 00-.469-.298c-.18-.07-.379-.106-.598-.106-.237 0-.447.041-.632.123-.185.082-.34.19-.465.325s-.22.29-.287.468c-.066.178-.1.363-.1.554zm11.416 4.621v-4.928c0-.215-.008-.39-.024-.527a1.2 1.2 0 00-.075-.325.348.348 0 00-.13-.167.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.488c-.107.203-.234.453-.38.749v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.239.467-.39.786v4.41h-1.093v-6.864h.91l.054 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.247.134-.052.283-.078.447-.078.37 0 .65.12.841.362.192.242.287.615.287 1.121.11-.237.217-.447.322-.632a2.31 2.31 0 01.338-.465c.12-.125.254-.22.4-.287.146-.066.314-.1.506-.1.861 0 1.292.664 1.292 1.99v4.997h-1.094zm7.212 0l-.027-.923c-.374.37-.754.636-1.139.8-.385.164-.79.246-1.213.246-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.396-.618 2.289 2.289 0 01-.127-.773c0-.688.256-1.227.77-1.617.512-.39 1.27-.584 2.272-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.643-.1c.223-.025.449-.037.677-.037.414 0 .788.045 1.12.137.333.09.615.23.845.417.23.186.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.058-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .132.021.259.062.38.041.12.108.227.199.32.09.094.21.168.355.223.146.055.324.082.533.082.274 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm5.756-2.755h-2.03v-.984h3.233v5.872h2.044v.991H108v-.991h2.256v-4.888zm.417-3.862c.132 0 .255.024.369.072a.893.893 0 01.297.201.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.202.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.664.893.893 0 01.298-.201.944.944 0 01.369-.072zm7.28 1.06h-2.03v-.978h3.233v8.668h2.044v.991h-5.503v-.991h2.256v-7.69zm8.06 6.528a1.113 1.113 0 01.803.335c.102.105.183.227.243.366.059.139.088.288.088.447 0 .155-.03.301-.088.438a1.148 1.148 0 01-.605.601c-.14.06-.286.09-.441.09-.16 0-.308-.03-.445-.09a1.148 1.148 0 01-.601-.601 1.088 1.088 0 01-.09-.438 1.178 1.178 0 01.332-.813 1.12 1.12 0 01.804-.335zm10.329 1.9c-.31.119-.628.206-.954.263a5.845 5.845 0 01-1.008.086c-1.085 0-1.92-.294-2.506-.882-.585-.588-.878-1.447-.878-2.577 0-.542.084-1.035.253-1.477.169-.442.406-.82.71-1.134.306-.315.67-.558 1.095-.728.424-.171.89-.257 1.401-.257.356 0 .688.025.998.075.31.05.606.133.889.247v1.134a3.77 3.77 0 00-.906-.338 4.248 4.248 0 00-.954-.106c-.305 0-.593.058-.864.174-.272.116-.51.284-.715.503a2.38 2.38 0 00-.485.8c-.119.314-.178.67-.178 1.066 0 .83.202 1.45.605 1.863.404.412.963.618 1.678.618a3.897 3.897 0 001.818-.438v1.108zm8.34-3.233a4.55 4.55 0 01-.226 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.38.263c-.493 0-.934-.076-1.323-.229a2.617 2.617 0 01-.991-.673 2.98 2.98 0 01-.623-1.1c-.143-.438-.215-.942-.215-1.511 0-.534.075-1.02.226-1.46.15-.44.367-.818.65-1.135a2.913 2.913 0 011.031-.738 3.447 3.447 0 011.381-.263c.492 0 .933.076 1.323.229.39.153.72.376.991.67.271.294.479.66.622 1.097.144.437.216.939.216 1.504zm-1.217.055c0-.424-.047-.795-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.633-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .423.047.795.14 1.114.094.319.227.584.4.796.173.212.383.372.629.479.246.107.524.16.834.16.355 0 .66-.07.912-.208.253-.14.46-.325.623-.557a2.4 2.4 0 00.355-.81 4.09 4.09 0 00.113-.974zm7.834 3.431v-4.928c0-.215-.008-.39-.024-.527a1.2 1.2 0 00-.075-.325.348.348 0 00-.13-.167.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.488 30.3 30.3 0 00-.38.749v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.24.467-.39.786v4.41h-1.093v-6.864h.909l.055 1.306c.118-.26.233-.481.345-.663.111-.183.227-.33.345-.441.119-.112.245-.194.38-.247.134-.052.283-.078.447-.078.37 0 .65.12.841.362.191.242.287.615.287 1.121.11-.237.217-.447.321-.632a2.31 2.31 0 01.339-.465c.12-.125.254-.22.4-.287.146-.066.314-.1.506-.1.86 0 1.292.664 1.292 1.99v4.997H151.3zm8.32-9.659l-4.69 11.129h-1.135l4.69-11.129h1.134zm7.074 9.66v-4.93c0-.214-.008-.39-.024-.526a1.2 1.2 0 00-.075-.325.348.348 0 00-.13-.167.43.43 0 00-.448.034c-.077.055-.16.144-.249.267a4.216 4.216 0 00-.294.488c-.107.203-.233.453-.38.749v4.41h-1.086v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.134-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.089.118-.188.28-.297.485-.11.205-.24.467-.39.786v4.41h-1.094v-6.864h.91l.054 1.306c.119-.26.234-.481.345-.663.112-.183.227-.33.346-.441.118-.112.245-.194.379-.247.134-.052.284-.078.448-.078.369 0 .65.12.84.362.192.242.288.615.288 1.121.109-.237.216-.447.321-.632a2.31 2.31 0 01.338-.465c.121-.125.254-.22.4-.287.146-.066.315-.1.506-.1.861 0 1.292.664 1.292 1.99v4.997h-1.094zm8.6-3.795c0 .169-.002.31-.007.424a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.201 0 .402-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.225.37.15.681.364.934.64.253.275.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.09-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.1-.4.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.657.048 12.3 12.3 0 01-1.281-.061 8.646 8.646 0 01-1.145-.199v-1.093c.4.114.8.2 1.196.26.397.059.79.088 1.183.088.57 0 .99-.077 1.264-.232.274-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.235-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.587-1.265c.215-.195.504-.358.869-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.895.044.328.03.67.081 1.026.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.068a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.503.275.677.435.173.159.297.339.372.54.075.2.113.428.113.683zm7.697 0a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.196.26.397.059.791.088 1.183.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.071.335.048.1.137.197.267.29.13.094.311.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.372.54.076.2.113.428.113.683zm6.672 1.873l-.027-.923c-.374.37-.753.636-1.138.8-.386.164-.79.246-1.214.246-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.396-.618 2.289 2.289 0 01-.127-.773c0-.688.257-1.227.77-1.617.512-.39 1.27-.584 2.272-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.189-.365-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.074a7.36 7.36 0 011.086-.3 7.6 7.6 0 01.643-.1c.223-.025.449-.037.677-.037.414 0 .788.045 1.12.137.334.09.615.23.845.417.23.186.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.214.058-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .132.02.259.062.38.04.12.107.227.198.32.091.094.21.168.355.223.146.055.324.082.534.082.273 0 .586-.083.94-.25.353-.166.725-.429 1.117-.789v-1.217zm8.36-2.768c.128.16.227.345.298.557.07.212.106.44.106.687 0 .355-.065.68-.195.974-.13.294-.313.546-.55.755a2.54 2.54 0 01-.851.489c-.33.116-.694.174-1.09.174-.288 0-.557-.03-.807-.092a2.064 2.064 0 01-.595-.229c-.087.128-.16.248-.219.362a.817.817 0 00-.089.383c0 .173.083.317.25.43.166.115.386.176.66.185l1.804.069c.342.009.658.052.947.13.29.077.538.189.745.335.207.145.37.325.485.54.117.214.175.462.175.745 0 .305-.066.594-.198.868a1.985 1.985 0 01-.612.721c-.276.207-.626.373-1.05.496-.423.123-.927.184-1.51.184-.556 0-1.03-.044-1.419-.133-.39-.089-.71-.212-.96-.37a1.49 1.49 0 01-.547-.56 1.51 1.51 0 01-.171-.714c0-.333.077-.624.232-.875.155-.25.395-.492.718-.725a1.133 1.133 0 01-.52-.475 1.322 1.322 0 01-.164-.632c0-.296.07-.567.21-.814.138-.246.303-.478.495-.697a3.57 3.57 0 01-.23-.307 1.91 1.91 0 01-.28-.701 2.536 2.536 0 01-.037-.462c0-.355.065-.68.195-.974.13-.294.312-.545.546-.755.235-.21.518-.373.848-.489.33-.116.696-.174 1.097-.174.169 0 .33.011.486.034.155.023.291.052.41.089h2.488v.97h-1.1zm-4.128 6.883c0 .324.168.56.505.708.338.148.807.222 1.409.222.378 0 .696-.034.953-.102.258-.069.465-.159.622-.27.158-.112.27-.24.339-.383.068-.144.102-.29.102-.441 0-.278-.114-.483-.342-.615-.227-.133-.576-.21-1.045-.233l-1.791-.061c-.15.1-.275.198-.373.294a1.292 1.292 0 00-.229.29 1.127 1.127 0 00-.116.294 1.309 1.309 0 00-.034.297zm.362-5.612c0 .219.036.42.11.602.072.182.175.337.307.465.132.127.288.226.468.297.18.07.38.106.598.106.237 0 .448-.04.633-.12.184-.08.34-.188.464-.324.126-.137.222-.294.288-.472.066-.178.099-.362.099-.554 0-.219-.037-.42-.11-.601a1.316 1.316 0 00-.307-.465 1.404 1.404 0 00-.469-.298c-.18-.07-.379-.106-.598-.106-.237 0-.447.041-.632.123-.185.082-.34.19-.465.325s-.22.29-.287.468c-.066.178-.1.363-.1.554zm12.318.827c0 .169-.002.31-.007.424a6.372 6.372 0 01-.02.321h-4.82c0 .702.197 1.241.588 1.617.392.376.958.564 1.696.564.2 0 .4-.008.601-.024.2-.016.395-.037.581-.065.187-.027.366-.058.537-.092.17-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.214.1-.399.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.177.84h3.609zm8.572 2.091a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.637 4.637 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.196.26.397.059.791.088 1.183.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488s.82-.195 1.367-.195c.269 0 .568.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.071.335.048.1.137.197.267.29.13.094.311.187.544.28.232.094.535.196.909.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.372.54.076.2.113.428.113.683zm5.804-7.786l-.198 3.206h-1.176l-.191-3.206h1.565z"/><path id=");" fill="#7E7C7B" d="M1.559 59.563c2.1 1.945 3.151 4.11 3.151 6.494 0 .492-.05 1.002-.15 1.53-.1.53-.272 1.07-.513 1.624a9.296 9.296 0 01-.974 1.685c-.408.57-.922 1.142-1.542 1.716l-.69-.704c.888-.88 1.552-1.798 1.99-2.755.437-.957.655-1.96.655-3.007 0-2.17-.881-4.12-2.645-5.852l.718-.731zm6.91 11.58c.233.009.459-.012.677-.062a2.02 2.02 0 00.578-.222c.166-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.099-.465 2.238 2.238 0 00-.222-.328 2.438 2.438 0 01-.222-.321.91.91 0 01-.1-.458.915.915 0 01.223-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.148.066.278.168.39.304.111.137.2.308.266.513.066.205.1.447.1.725 0 .378-.07.741-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.483-1.032.643-.413.16-.892.24-1.44.24v-.937zm2.14-8.388c.142 0 .276.028.404.085a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.23.332 1.058 1.058 0 01-.738.308c-.145 0-.281-.028-.406-.083a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.083-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.332-.23.971.971 0 01.406-.085z"/><path id="let" fill="#1C85B5" d="M2.495 117.06H.465v-.978h3.233v8.668h2.044v.991H.24v-.991h2.256v-7.69zm11.156 4.887c0 .169-.002.31-.006.424a6.37 6.37 0 01-.021.321h-4.82c0 .702.197 1.241.589 1.617.392.376.957.564 1.695.564.2 0 .4-.008.601-.024.201-.016.395-.037.582-.065.186-.027.365-.058.536-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.069-.94.208-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.575 1.005.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.571-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.71.15-.215.1-.4.242-.555.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.177.84h3.61zm8.668 3.869a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.861 0-1.504-.195-1.928-.585-.424-.39-.635-.987-.635-1.794v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.131.86.393 1.104.262.244.648.366 1.159.366.219 0 .458-.017.718-.051.26-.034.53-.088.813-.16v1.025z"/><path id="messages" fill="#181717" d="M35.84 125.741v-4.928c0-.215-.007-.39-.023-.527a1.2 1.2 0 00-.075-.325.348.348 0 00-.13-.167.43.43 0 00-.448.034c-.077.055-.16.144-.25.267a4.216 4.216 0 00-.293.488c-.107.203-.234.453-.38.749v4.41h-1.087v-4.8c0-.25-.008-.453-.024-.608a1.42 1.42 0 00-.075-.362.34.34 0 00-.133-.178.431.431 0 00-.43.02.988.988 0 00-.243.247c-.09.118-.188.28-.298.485-.11.205-.24.467-.39.786v4.41h-1.093v-6.864h.909l.055 1.306c.118-.26.233-.481.345-.663.112-.183.227-.33.345-.441.119-.112.245-.194.38-.247.134-.052.283-.078.447-.078.37 0 .65.12.841.362.191.242.287.615.287 1.121.11-.237.217-.447.321-.632a2.31 2.31 0 01.339-.465c.12-.125.254-.22.4-.287.146-.066.314-.1.506-.1.86 0 1.292.664 1.292 1.99v4.997H35.84zm8.6-3.794c0 .169-.002.31-.006.424a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.187-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.194-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.585.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.575 1.005.13.394.194.835.194 1.322zm-1.237-.17a2.593 2.593 0 00-.089-.838 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.1-.4.242-.555.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.177.84h3.61zm8.572 2.091a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.197.26.396.059.79.088 1.182.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.45-.56 1.649 1.649 0 01-.165-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.367-.195.27 0 .568.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.373.54.075.2.112.428.112.683zm7.698 0a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.493.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.402.114.8.2 1.197.26.396.059.79.088 1.182.088.57 0 .992-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.368-.195.268 0 .567.015.895.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.296 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.232.094.535.196.91.305.405.118.747.243 1.025.372.278.13.503.275.676.435.173.159.298.339.373.54.075.2.113.428.113.683zm6.672 1.873l-.028-.923c-.374.37-.753.636-1.138.8-.385.164-.79.246-1.213.246-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.397-.618 2.289 2.289 0 01-.126-.773c0-.688.256-1.227.769-1.617.513-.39 1.27-.584 2.273-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.387 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.642-.1c.224-.025.45-.037.677-.037.415 0 .789.045 1.121.137.333.09.614.23.845.417.23.186.406.421.53.704.122.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.511c-.296 0-.552.03-.766.09-.214.058-.39.143-.526.252-.137.11-.238.24-.304.393-.066.153-.1.325-.1.516 0 .132.021.259.062.38.041.12.107.227.198.32.092.094.21.168.356.223.146.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm8.36-2.768c.128.16.227.345.297.557.071.212.106.44.106.687 0 .355-.065.68-.195.974-.13.294-.313.546-.55.755a2.54 2.54 0 01-.85.489c-.331.116-.695.174-1.091.174-.287 0-.556-.03-.807-.092a2.064 2.064 0 01-.595-.229c-.086.128-.16.248-.218.362a.817.817 0 00-.09.383c0 .173.084.317.25.43.167.115.387.176.66.185l1.805.069c.341.009.657.052.946.13.29.077.538.189.746.335.207.145.369.325.485.54.116.214.174.462.174.745 0 .305-.066.594-.198.868a1.985 1.985 0 01-.612.721c-.276.207-.625.373-1.05.496-.423.123-.927.184-1.51.184-.556 0-1.029-.044-1.418-.133-.39-.089-.71-.212-.96-.37a1.49 1.49 0 01-.548-.56 1.51 1.51 0 01-.17-.714c0-.333.077-.624.232-.875.155-.25.394-.492.718-.725a1.133 1.133 0 01-.52-.475 1.322 1.322 0 01-.164-.632c0-.296.07-.567.209-.814.139-.246.304-.478.495-.697a3.57 3.57 0 01-.229-.307 1.91 1.91 0 01-.28-.701 2.536 2.536 0 01-.038-.462c0-.355.065-.68.195-.974.13-.294.312-.545.547-.755.235-.21.517-.373.848-.489.33-.116.696-.174 1.097-.174.168 0 .33.011.485.034.155.023.292.052.41.089h2.489v.97h-1.101zm-4.129 6.883c0 .324.169.56.506.708.337.148.807.222 1.408.222.378 0 .696-.034.954-.102.257-.069.465-.159.622-.27.157-.112.27-.24.338-.383.069-.144.103-.29.103-.441 0-.278-.114-.483-.342-.615-.228-.133-.577-.21-1.046-.233l-1.791-.061c-.15.1-.275.198-.373.294a1.292 1.292 0 00-.229.29 1.127 1.127 0 00-.116.294 1.309 1.309 0 00-.034.297zm.362-5.612c0 .219.037.42.11.602.073.182.175.337.307.465.133.127.289.226.469.297.18.07.379.106.598.106.237 0 .448-.04.632-.12.185-.08.34-.188.465-.324.125-.137.221-.294.287-.472.066-.178.1-.362.1-.554 0-.219-.037-.42-.11-.601a1.316 1.316 0 00-.308-.465 1.404 1.404 0 00-.468-.298c-.18-.07-.38-.106-.598-.106-.237 0-.448.041-.633.123-.184.082-.339.19-.464.325-.126.135-.221.29-.287.468-.067.178-.1.363-.1.554zm12.319.827c0 .169-.003.31-.007.424a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.367-.058.538-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.589 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.585.98-.776a2.94 2.94 0 011.32-.287c.479 0 .902.075 1.272.225.369.15.68.364.933.64.253.275.444.61.574 1.005.13.394.195.835.195 1.322zm-1.238-.17a2.593 2.593 0 00-.088-.838 1.763 1.763 0 00-.339-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.496.05-.71.15-.215.1-.4.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61zm8.573 2.091a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.599.259c-.212.068-.43.119-.652.15a4.638 4.638 0 01-.657.048 12.3 12.3 0 01-1.281-.061 8.646 8.646 0 01-1.145-.199v-1.093c.4.114.8.2 1.196.26.396.059.79.088 1.182.088.57 0 .992-.077 1.265-.232.274-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.452-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.504-.358.868-.488.365-.13.82-.195 1.368-.195.269 0 .567.015.895.044.328.03.67.081 1.026.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.232.094.536.196.91.305.405.118.747.243 1.025.372.278.13.503.275.676.435.174.159.298.339.373.54.075.2.113.428.113.683z"/><path id="=" fill="#DBAF88" d="M105.916 121.325h-5.879v-1.005h5.879v1.005zm0 2.38h-5.879v-1.006h5.879v1.005z"/><path id="await" fill="#1C85B5" d="M120.025 125.741l-.027-.923c-.374.37-.753.636-1.138.8-.385.164-.79.246-1.214.246-.391 0-.726-.05-1.004-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.397-.618 2.289 2.289 0 01-.126-.773c0-.688.256-1.227.769-1.617.512-.39 1.27-.584 2.273-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.387 0-.769.043-1.145.13-.376.086-.764.21-1.165.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.642-.1c.224-.025.45-.037.677-.037.415 0 .789.045 1.121.137.333.09.614.23.844.417.23.186.407.421.53.704.123.282.185.615.185.998v4.73h-1.067zm-.13-3.124h-1.51c-.296 0-.552.03-.766.09-.214.058-.39.143-.526.252-.137.11-.238.24-.304.393-.067.153-.1.325-.1.516 0 .132.021.259.062.38.041.12.107.227.198.32.091.094.21.168.356.223.146.055.323.082.533.082.273 0 .587-.083.94-.25.353-.166.726-.429 1.118-.789v-1.217zm9.776-3.74l-.998 6.864h-1.443l-.99-2.87-.199-.698-.226.738-.95 2.83h-1.401l-.991-6.863h1.162l.574 4.662.123 1.04.294-.91.998-3.083h.855l1.073 3.042.307.91.103-.965.533-4.696h1.176zm5.749 6.864l-.027-.923c-.374.37-.754.636-1.139.8-.385.164-.79.246-1.213.246-.392 0-.727-.05-1.005-.15a1.883 1.883 0 01-.687-.414 1.603 1.603 0 01-.396-.618 2.289 2.289 0 01-.127-.773c0-.688.257-1.227.77-1.617.512-.39 1.27-.584 2.272-.584h1.422v-.602c0-.405-.13-.73-.39-.974-.26-.243-.656-.365-1.19-.365-.386 0-.768.043-1.144.13-.376.086-.765.21-1.166.369v-1.074a7.36 7.36 0 011.087-.3 7.6 7.6 0 01.643-.1c.223-.025.449-.037.677-.037.414 0 .788.045 1.12.137.333.09.615.23.845.417.23.186.407.421.53.704.123.282.184.615.184.998v4.73h-1.066zm-.13-3.124h-1.51c-.297 0-.552.03-.766.09-.215.058-.39.143-.527.252-.136.11-.238.24-.304.393-.066.153-.099.325-.099.516 0 .132.02.259.062.38.04.12.107.227.198.32.09.094.21.168.355.223.146.055.324.082.533.082.274 0 .587-.083.94-.25.354-.166.726-.429 1.118-.789v-1.217zm5.756-2.755h-2.03v-.984h3.233v5.872h2.044v.991h-5.503v-.991h2.256v-4.888zm.417-3.862c.132 0 .255.024.369.072a.893.893 0 01.297.201.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.202.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.96.96 0 01-.27-.663.919.919 0 01.27-.664.893.893 0 01.298-.201.944.944 0 01.369-.072zm10.473 9.646a5.793 5.793 0 01-.834.146 8.52 8.52 0 01-.875.045c-.862 0-1.504-.195-1.928-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.92v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.13.86.393 1.104.262.244.648.366 1.158.366.219 0 .458-.017.718-.051.26-.034.531-.088.814-.16v1.025z"/><path id="response" fill="#181717" d="M162.08 118.878h1.087l.034 1.265c.406-.488.806-.841 1.2-1.06a2.436 2.436 0 011.193-.328c.71 0 1.25.23 1.616.69.367.46.537 1.144.51 2.051h-1.203c.013-.601-.074-1.038-.264-1.309-.189-.271-.466-.407-.83-.407-.16 0-.32.029-.482.086a1.947 1.947 0 00-.5.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm13.214 3.07c0 .168-.002.31-.007.423a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.201 0 .402-.008.602-.024.2-.016.394-.037.581-.065.187-.027.366-.058.537-.092.17-.034.329-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.518-.24a2.597 2.597 0 01-1.042-.694 2.857 2.857 0 01-.602-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.209-1.37a3.44 3.44 0 01.608-1.135c.267-.326.594-.585.981-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.27.225.37.15.681.364.934.64.253.275.444.61.574 1.005.13.394.195.835.195 1.322zm-1.237-.172a2.593 2.593 0 00-.09-.837 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.793-.154c-.26 0-.497.05-.711.15-.215.1-.4.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61zm8.572 2.092a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.657.048 12.3 12.3 0 01-1.281-.061 8.646 8.646 0 01-1.145-.199v-1.093c.4.114.8.2 1.196.26.397.059.79.088 1.183.088.57 0 .99-.077 1.264-.232.274-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.235-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.587-1.265c.215-.195.504-.358.869-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.895.044.328.03.67.081 1.026.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.068a1.553 1.553 0 00-.493.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.503.275.677.435.173.159.297.339.372.54.075.2.113.428.113.683zm8.128-1.681c0 .61-.086 1.143-.257 1.6-.17.455-.406.833-.707 1.134a2.9 2.9 0 01-1.066.677c-.41.15-.855.225-1.333.225-.22 0-.437-.011-.653-.034a4.952 4.952 0 01-.66-.116v2.87h-1.19v-9.665h1.06l.075 1.148c.342-.469.707-.798 1.094-.987.387-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.296.164.545.395.745.694.2.298.351.658.451 1.08.1.421.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.079-.991a2.586 2.586 0 00-.25-.772 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.635-.181c-.15 0-.303.024-.458.072a1.857 1.857 0 00-.482.239 3.59 3.59 0 00-.527.444 7 7 0 00-.59.687v3.33c.218.09.448.162.69.215.241.052.478.078.71.078.643 0 1.147-.217 1.511-.652.365-.436.547-1.09.547-1.966zm9.023.014a4.55 4.55 0 01-.225 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.38.263c-.493 0-.934-.076-1.323-.229a2.617 2.617 0 01-.992-.673 2.98 2.98 0 01-.622-1.1c-.143-.438-.215-.942-.215-1.511 0-.534.075-1.02.226-1.46.15-.44.366-.818.649-1.135a2.913 2.913 0 011.032-.738 3.447 3.447 0 011.381-.263c.492 0 .933.076 1.323.229.39.153.72.376.991.67.271.294.479.66.622 1.097.144.437.215.939.215 1.504zm-1.216.055c0-.424-.047-.795-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.633-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.355.81 4.06 4.06 0 00-.113.968c0 .423.047.795.14 1.114.093.319.227.584.4.796.173.212.383.372.629.479.246.107.524.16.834.16.355 0 .66-.07.912-.208.253-.14.46-.325.622-.557a2.4 2.4 0 00.356-.81 4.09 4.09 0 00.113-.974zm2.94-3.432h1.059l.048 1.107c.2-.237.394-.434.58-.591.187-.157.37-.284.55-.38a2.06 2.06 0 01.551-.201c.187-.039.38-.058.581-.058.707 0 1.241.208 1.603.625.363.417.544 1.045.544 1.884v4.477h-1.19v-4.382c0-.537-.1-.935-.3-1.192-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.49.434-.175.182-.372.41-.59.683v4.553h-1.19v-6.863zm13.131 4.99a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.4.114.8.2 1.196.26.397.059.79.088 1.183.088.57 0 .99-.077 1.264-.232.274-.155.41-.376.41-.663a.812.812 0 00-.064-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.895.044.329.03.67.081 1.026.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.546.023-.748.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.083.346c0 .123.024.235.072.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.503.275.677.435.173.159.297.339.372.54.075.2.113.428.113.683zm8.06-1.92c0 .168-.003.31-.007.423a6.37 6.37 0 01-.02.321h-4.82c0 .702.196 1.241.588 1.617.392.376.957.564 1.695.564.2 0 .401-.008.602-.024.2-.016.394-.037.58-.065.188-.027.366-.058.537-.092.171-.034.33-.072.475-.113v.978a9.161 9.161 0 01-2.365.308c-.588 0-1.094-.08-1.517-.24a2.597 2.597 0 01-1.043-.694 2.857 2.857 0 01-.601-1.114 5.273 5.273 0 01-.195-1.494c0-.483.07-.94.208-1.37a3.44 3.44 0 01.609-1.135c.266-.326.593-.585.98-.776a2.94 2.94 0 011.32-.287c.478 0 .902.075 1.271.225.37.15.68.364.933.64.253.275.445.61.575 1.005.13.394.195.835.195 1.322zm-1.238-.172a2.593 2.593 0 00-.089-.837 1.763 1.763 0 00-.338-.653 1.576 1.576 0 00-.57-.427 1.916 1.916 0 00-.794-.154c-.26 0-.497.05-.71.15-.215.1-.4.242-.554.424a2.19 2.19 0 00-.376.657 3.026 3.026 0 00-.178.84h3.61z"/><path id="." fill="#7E7C7B" d="M226.078 123.588a1.113 1.113 0 01.803.335c.103.105.184.227.243.366.06.139.089.288.089.447 0 .155-.03.301-.089.438a1.148 1.148 0 01-.605.601c-.139.06-.286.09-.44.09-.16 0-.308-.03-.445-.09a1.148 1.148 0 01-.602-.601 1.088 1.088 0 01-.089-.438 1.178 1.178 0 01.332-.813 1.12 1.12 0 01.803-.335z"/><path id="json" fill="#181717" d="M235.724 118.878v6.74c0 .479-.069.904-.205 1.275a2.557 2.557 0 01-.588.94 2.485 2.485 0 01-.94.581c-.372.132-.79.198-1.255.198a5.24 5.24 0 01-.987-.092 4.354 4.354 0 01-.872-.256v-1.135a4.512 4.512 0 001.976.472c.52 0 .927-.164 1.223-.493.296-.328.445-.81.445-1.449v-5.797h-3.364v-.984h4.567zm-.718-2.878c.132 0 .255.024.369.072a.893.893 0 01.297.201.96.96 0 01.27.664.919.919 0 01-.27.663.893.893 0 01-.297.201.944.944 0 01-.37.072.944.944 0 01-.368-.072.893.893 0 01-.298-.201.919.919 0 01-.27-.663.96.96 0 01.27-.663.893.893 0 01.298-.202.944.944 0 01.369-.072zm9.201 7.868a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.4.114.8.2 1.196.26.397.059.79.088 1.183.088.57 0 .991-.077 1.264-.232.274-.155.41-.376.41-.663a.812.812 0 00-.064-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.981-.328 6.963 6.963 0 01-.858-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.451-.56 1.649 1.649 0 01-.164-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488.364-.13.82-.195 1.367-.195.269 0 .567.015.895.044.329.03.67.081 1.026.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.872-.065c-.296 0-.545.023-.748.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.023.235.071.335.048.1.137.197.267.29.13.094.31.187.543.28.233.094.536.196.91.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.372.54.075.2.113.428.113.683zm8.237-1.613a4.55 4.55 0 01-.225 1.466c-.15.445-.367.825-.65 1.142a2.913 2.913 0 01-1.032.738 3.447 3.447 0 01-1.38.263c-.493 0-.934-.076-1.324-.229a2.617 2.617 0 01-.99-.673 2.98 2.98 0 01-.623-1.1c-.143-.438-.215-.942-.215-1.511 0-.534.075-1.02.225-1.46.15-.44.367-.818.65-1.135a2.913 2.913 0 011.032-.738 3.447 3.447 0 011.381-.263c.492 0 .933.076 1.323.229.39.153.72.376.99.67.272.294.48.66.623 1.097.144.437.215.939.215 1.504zm-1.216.055c0-.424-.047-.795-.14-1.111a2.237 2.237 0 00-.4-.793 1.65 1.65 0 00-.633-.479 2.077 2.077 0 00-.83-.16c-.356 0-.66.07-.913.208-.253.14-.46.325-.622.557a2.4 2.4 0 00-.356.81 4.06 4.06 0 00-.112.968c0 .423.046.795.14 1.114.093.319.227.584.4.796.173.212.383.372.629.479.246.107.524.16.834.16.355 0 .66-.07.912-.208.253-.14.46-.325.622-.557a2.4 2.4 0 00.356-.81 4.09 4.09 0 00.113-.974zm2.939-3.432h1.06l.047 1.107c.2-.237.395-.434.581-.591.187-.157.37-.284.55-.38a2.06 2.06 0 01.551-.201c.187-.039.38-.058.581-.058.706 0 1.24.208 1.603.625.362.417.544 1.045.544 1.884v4.477h-1.19v-4.382c0-.537-.1-.935-.3-1.192-.201-.258-.5-.387-.896-.387-.146 0-.288.022-.427.065a1.557 1.557 0 00-.434.226c-.15.107-.314.252-.49.434-.175.182-.372.41-.59.683v4.553h-1.19v-6.863z"/><path id="();" fill="#7E7C7B" d="M265.973 128.612c-2.101-1.946-3.152-4.097-3.152-6.453 0-.551.056-1.102.168-1.65.111-.55.291-1.1.54-1.652a9.028 9.028 0 01.984-1.654c.408-.551.904-1.098 1.487-1.64l.69.704c-1.768 1.745-2.652 3.68-2.652 5.803 0 1.058.223 2.074.67 3.05.447.974 1.107 1.895 1.982 2.76l-.717.732zm4.99-13.05c2.1 1.946 3.151 4.111 3.151 6.495 0 .492-.05 1.002-.15 1.53-.1.53-.271 1.07-.513 1.624a9.296 9.296 0 01-.974 1.685c-.408.57-.922 1.142-1.541 1.716l-.69-.704c.888-.88 1.55-1.798 1.988-2.755.438-.957.657-1.96.657-3.007 0-2.17-.882-4.12-2.646-5.852l.718-.731zm6.911 11.58c.232.01.458-.01.677-.06a2.02 2.02 0 00.577-.223c.167-.098.3-.22.4-.366a.86.86 0 00.15-.499.95.95 0 00-.098-.465 2.238 2.238 0 00-.223-.328 2.438 2.438 0 01-.222-.321.91.91 0 01-.099-.458.915.915 0 01.222-.588.857.857 0 01.287-.212 1.12 1.12 0 01.875.017c.149.066.278.168.39.304.112.137.2.308.267.513.066.205.099.447.099.725 0 .378-.07.741-.209 1.09a2.657 2.657 0 01-.622.926c-.276.27-.62.483-1.032.643-.412.16-.892.24-1.439.24v-.937zm2.14-8.387c.14 0 .275.028.403.085a1.107 1.107 0 01.564.564.978.978 0 01.085.404.971.971 0 01-.085.406 1.12 1.12 0 01-.229.332 1.058 1.058 0 01-.738.308c-.146 0-.282-.028-.407-.083a1.07 1.07 0 01-.557-.557 1.006 1.006 0 01-.082-.406 1.058 1.058 0 01.308-.738 1.12 1.12 0 01.33-.23.971.971 0 01.408-.085z"/><path id="..." fill="#DBAF88" d="M2.857 137.588a1.113 1.113 0 01.804.335c.102.105.183.227.242.366.06.139.09.288.09.447 0 .155-.03.301-.09.438a1.148 1.148 0 01-.605.601c-.139.06-.286.09-.44.09-.16 0-.308-.03-.445-.09a1.148 1.148 0 01-.601-.601 1.088 1.088 0 01-.09-.438 1.178 1.178 0 01.332-.813 1.12 1.12 0 01.803-.335zm7.698 0a1.113 1.113 0 01.803.335c.102.105.183.227.243.366.059.139.088.288.088.447 0 .155-.03.301-.088.438a1.148 1.148 0 01-.605.601c-.14.06-.286.09-.441.09-.16 0-.308-.03-.445-.09a1.148 1.148 0 01-.601-.601 1.088 1.088 0 01-.09-.438 1.178 1.178 0 01.332-.813 1.12 1.12 0 01.804-.335zm7.697 0a1.113 1.113 0 01.803.335c.103.105.184.227.243.366.06.139.089.288.089.447 0 .155-.03.301-.09.438a1.148 1.148 0 01-.604.601c-.14.06-.286.09-.441.09-.16 0-.308-.03-.444-.09a1.148 1.148 0 01-.602-.601 1.088 1.088 0 01-.089-.438 1.178 1.178 0 01.332-.813 1.12 1.12 0 01.803-.335z"/><path id="</script>" fill="#7E7C7B" d="M5.339 171.099l-.745.752L0 168.016l4.594-3.835.745.758-3.726 3.063 3.726 3.097zm8.032-9.017l-4.69 11.129H7.548l4.69-11.129h1.134zm7.615 7.786a1.634 1.634 0 01-.458 1.159 2.084 2.084 0 01-.492.38 3.272 3.272 0 01-.598.259c-.212.068-.43.119-.653.15a4.638 4.638 0 01-.656.048 12.3 12.3 0 01-1.282-.061 8.646 8.646 0 01-1.145-.199v-1.093c.401.114.8.2 1.196.26.397.059.791.088 1.183.088.57 0 .991-.077 1.265-.232.273-.155.41-.376.41-.663a.812.812 0 00-.065-.332.753.753 0 00-.236-.28 2.254 2.254 0 00-.53-.277 11.81 11.81 0 00-.98-.328 6.963 6.963 0 01-.859-.311 2.877 2.877 0 01-.68-.414 1.776 1.776 0 01-.45-.56 1.649 1.649 0 01-.165-.759 1.744 1.744 0 01.588-1.265c.214-.195.503-.358.868-.488s.82-.195 1.367-.195c.27 0 .568.015.896.044.328.03.67.081 1.025.154v1.06a9.803 9.803 0 00-1.063-.202 6.926 6.926 0 00-.871-.065c-.297 0-.546.023-.749.068a1.553 1.553 0 00-.492.188.734.734 0 00-.27.28.748.748 0 00-.082.346c0 .123.024.235.072.335.047.1.136.197.266.29.13.094.311.187.544.28.232.094.535.196.909.305.405.118.747.243 1.025.372.278.13.504.275.677.435.173.159.297.339.373.54.075.2.112.428.112.683zm7.595 1.62c-.31.119-.628.206-.954.263a5.845 5.845 0 01-1.008.086c-1.085 0-1.92-.294-2.505-.882-.586-.588-.879-1.447-.879-2.577 0-.542.085-1.035.253-1.477a3.142 3.142 0 011.805-1.863c.424-.17.89-.256 1.401-.256.356 0 .688.025.998.075.31.05.607.133.89.247v1.134a3.771 3.771 0 00-.907-.338 4.248 4.248 0 00-.953-.106c-.306 0-.594.058-.865.174-.271.116-.51.284-.714.503a2.38 2.38 0 00-.486.8c-.118.314-.178.67-.178 1.066 0 .83.202 1.45.605 1.863.404.412.963.618 1.679.618a3.897 3.897 0 001.818-.438v1.108zm2.646-6.61h1.086l.035 1.265c.405-.488.805-.841 1.2-1.06a2.436 2.436 0 011.192-.328c.711 0 1.25.23 1.617.69.367.46.537 1.144.51 2.051h-1.204c.014-.601-.074-1.038-.263-1.309-.19-.271-.466-.407-.83-.407-.16 0-.32.029-.483.086a1.947 1.947 0 00-.499.273 4.47 4.47 0 00-.543.482 8.898 8.898 0 00-.615.711v4.41h-1.203v-6.864zm9.754.984h-2.03v-.984h3.234v5.872h2.044v.991h-5.503v-.991h2.255v-4.888zm.417-3.862c.133 0 .256.024.37.072a.893.893 0 01.297.201.919.919 0 01.27.663.96.96 0 01-.27.663.893.893 0 01-.297.202.944.944 0 01-.37.072.944.944 0 01-.369-.072.893.893 0 01-.297-.201.96.96 0 01-.27-.663.919.919 0 01.27-.664.893.893 0 01.297-.201.944.944 0 01.37-.072zm10.808 6.187c0 .61-.085 1.143-.256 1.6-.171.455-.407.833-.708 1.134a2.9 2.9 0 01-1.066.677c-.41.15-.855.225-1.333.225-.219 0-.437-.011-.653-.034a4.952 4.952 0 01-.66-.116v2.87h-1.19v-9.665h1.06l.076 1.148c.341-.469.706-.798 1.093-.987.388-.19.807-.284 1.258-.284.392 0 .736.082 1.032.246.297.164.545.395.745.694.201.298.351.658.452 1.08.1.421.15.892.15 1.412zm-1.217.054c0-.36-.026-.69-.078-.991a2.586 2.586 0 00-.25-.772 1.37 1.37 0 00-.437-.503 1.104 1.104 0 00-.636-.181c-.15 0-.303.024-.458.072a1.857 1.857 0 00-.482.239 3.59 3.59 0 00-.526.444 7 7 0 00-.592.687v3.33c.219.09.45.162.69.215.242.052.48.078.712.078.642 0 1.146-.217 1.51-.652.365-.436.547-1.09.547-1.966zm8.58 3.405a5.793 5.793 0 01-.835.146 8.52 8.52 0 01-.875.045c-.861 0-1.504-.195-1.927-.585-.424-.39-.636-.987-.636-1.794v-3.582h-1.921v-.998h1.92v-1.887l1.19-.307v2.194h3.083v.998h-3.083v3.486c0 .492.131.86.393 1.104.262.244.649.366 1.16.366.218 0 .457-.017.717-.051.26-.034.53-.088.813-.16v1.025zm2.501-6.707l.739-.758 4.593 3.835-4.593 3.835-.739-.752 3.726-3.07-3.726-3.09z"/></g><text id="evil.com" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="129" y="36">evil.com</tspan></text><path id="Rectangle-1-Copy" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M494 48h140v199H494z"/><text id="got-the-cookie?-okay" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="508.465" y="133">got the cookie?</tspan> <tspan x="544.852" y="155">okay!</tspan></text><text id="gmail.com" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="16" font-weight="bold"><tspan x="528" y="36">gmail.com</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M308.215 100.812l1.487.202 158.667 21.513 1.075-7.927 17.551 11.967-20.104 6.861 1.075-7.928-158.668-21.514-1.486-.201.403-2.973z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M491.796 151.82l.383 2.976-1.488.192L332.54 175.34l1.021 7.935-20.057-6.997 17.632-11.847 1.021 7.934 158.152-20.354 1.487-.191z"/><text id="GET-/messages" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" transform="rotate(7 408.585 101.617)"><tspan x="352.085" y="107.617">GET /messages</tspan></text><text id="cookie:-user=John" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" transform="rotate(7 402.573 127.8)"><tspan x="336.073" y="133.8">cookie: user=John</tspan></text><text id="{"messages":-[...]}" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" transform="rotate(-7 408.333 176.276)"><tspan x="342.833" y="182.276">{"messages": [...]}</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:5-network/05-fetch-crossorigin/cors-gmail-messages.svg diff --git a/5-network/04-fetch-crossorigin/xhr-another-domain.svg b/5-network/05-fetch-crossorigin/xhr-another-domain.svg similarity index 55% rename from 5-network/04-fetch-crossorigin/xhr-another-domain.svg rename to 5-network/05-fetch-crossorigin/xhr-another-domain.svg index 3525652bc..47bf429ed 100644 --- a/5-network/04-fetch-crossorigin/xhr-another-domain.svg +++ b/5-network/05-fetch-crossorigin/xhr-another-domain.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:5-network/04-fetch-crossorigin/xhr-another-domain.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="610px" height="411px" viewBox="0 0 610 411" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -48,4 +49,7 @@ <path id="Line-3" d="M85,332 L302,332 L302,334 L85,334 L85,340 L71,333 L85,326 L85,332 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="633" height="411" viewBox="0 0 633 411"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="xhr-another-domain.svg"><path id="Rectangle-227" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 16h128v64H1z"/><text id="JavaScript" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="35" y="50">JavaScript</tspan></text><path id="Rectangle-228" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M261 16h128v64H261z"/><text id="Browser" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="296" y="50">Browser</tspan></text><path id="Rectangle-229" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M504 16h128v64H504z"/><text id="Server" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="544" y="50">Server</tspan></text><path id="Line" stroke="#7E7C7B" stroke-linecap="square" d="M66 81v320M326 81v320M569 81v320"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M312 133l14 7-14 7v-6H66v-2h246v-6z"/><text id="Origin:-https://java" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="354.313" y="205">Origin: https://javascript.info</tspan></text><text id="HTTP-request" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="401.015" y="174">HTTP-request</tspan></text><text id="fetch()" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="171" y="131">fetch()</tspan></text><text id="HTTP-response" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="398.419" y="250">HTTP-response</tspan></text><text id="Access-Control-Allow" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="347" y="287">Access-Control-Allow-Origin: * </tspan> <tspan x="364.141" y="305">(or https://javascript.info)</tspan></text><text id="if-the-header-allows" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="85" y="325">if the header allows, then success,</tspan></text><text id="otherwise-fail" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="148" y="353">otherwise fail</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M554 180l14 7-14 7v-6H328v-2h226v-6z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M340 258v6h228v2H340v6l-14-7 14-7z"/><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M80 326v6h247v2H80v6l-14-7 14-7z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:5-network/05-fetch-crossorigin/xhr-another-domain.svg diff --git a/5-network/04-fetch-crossorigin/xhr-preflight.svg b/5-network/05-fetch-crossorigin/xhr-preflight.svg similarity index 54% rename from 5-network/04-fetch-crossorigin/xhr-preflight.svg rename to 5-network/05-fetch-crossorigin/xhr-preflight.svg index 5ec17688b..0546c805a 100644 --- a/5-network/04-fetch-crossorigin/xhr-preflight.svg +++ b/5-network/05-fetch-crossorigin/xhr-preflight.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:5-network/04-fetch-crossorigin/xhr-preflight.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="610px" height="633px" viewBox="0 0 610 633" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -79,4 +80,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="620" height="633" viewBox="0 0 620 633"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="xhr-preflight.svg"><path id="Rectangle-227" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M1 16h128v64H1z"/><text id="JavaScript" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="33" y="50">JavaScript</tspan></text><path id="Rectangle-228" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M240 16h128v64H240z"/><text id="Browser" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="278" y="48">Browser</tspan></text><path id="Rectangle-229" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M491 16h128v64H491z"/><text id="Server" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="534" y="50">Server</tspan></text><path id="Line" stroke="#7E7C7B" stroke-linecap="square" d="M66 82v520M305 82v520M556 82v520"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M291 133l14 7-14 7-.001-6H67v-2h223.999l.001-6z"/><text id="fetch()" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="166" y="131">fetch()</tspan></text><text id="OPTIONS" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="400" y="173">OPTIONS</tspan></text><text id="Origin-Access-Contro" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="406.721" y="198">Origin</tspan> <tspan x="321.866" y="216">Access-Control-Request-Method</tspan> <tspan x="320" y="234">Access-Control-Request-Headers</tspan></text><path id="Line" fill="#C06334" fill-rule="nonzero" d="M319 285.5v6h237v2H319v6l-14-7 14-7z"/><text id="200-OK" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="407" y="285">200 OK</tspan></text><path id="Line-4" fill="#C06334" fill-rule="nonzero" d="M542 173l14 7-14 7-.001-6H305v-2h236.999l.001-6z"/><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M319 495v6h237v2H319v6l-14-7 14-7z"/><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M90.5 544.5v8h214v3h-214v8l-19-9.5 19-9.5z"/><text id="Access-Control-Allow" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="336" y="522">Access-Control-Allow-Origin</tspan></text><text id="Main-HTTP-response" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="355" y="495">Main HTTP-response</tspan></text><text id="otherwise-error" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="130" y="570">otherwise error</tspan></text><text id="if-allowed:-success," fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="115.5" y="545">if allowed: success,</tspan></text><path id="Line-5" fill="#C06334" fill-rule="nonzero" d="M542 429l14 7-14 7-.001-6H304v-2h237.999l.001-6z"/><text id="Origin" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="405" y="454">Origin</tspan></text><text id="Main-HTTP-request" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="360" y="429">Main HTTP-request</tspan></text><text id="preflight" fill="#7E7C7B" font-family="OpenSans-Regular, Open Sans" font-size="22" font-weight="normal"><tspan x="210.145" y="186">preflight</tspan></text><text id="if-allowed" fill="#7E7C7B" font-family="OpenSans-Regular, Open Sans" font-size="22" font-weight="normal"><tspan x="198.338" y="441">if allowed</tspan></text><text id="1" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="570.139" y="186">1</tspan></text><text id="2" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="570.139" y="302">2</tspan></text><text id="3" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="570.139" y="443">3</tspan></text><text id="4" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal"><tspan x="570.139" y="509">4</tspan></text><text id="Access-Control-Allow" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"> <tspan x="340.89" y="311.33">Access-Control-Allow-Origin</tspan> <tspan x="332" y="329.33">Access-Control-Allow-Methods</tspan> <tspan x="333.473" y="347.33">Access-Control-Allow-Headers</tspan> <tspan x="352.443" y="365.33">Access-Control-Max-Age</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:5-network/05-fetch-crossorigin/xhr-preflight.svg diff --git a/5-network/06-fetch-api/article.md b/5-network/06-fetch-api/article.md new file mode 100644 index 000000000..5f55c78ef --- /dev/null +++ b/5-network/06-fetch-api/article.md @@ -0,0 +1,224 @@ + +# Fetch API + +So far, we know quite a bit about `fetch`. + +Let's see the rest of API, to cover all its abilities. + +```smart +Please note: most of these options are used rarely. You may skip this chapter and still use `fetch` well. + +Still, it's good to know what `fetch` can do, so if the need arises, you can return and read the details. +``` + +Here's the full list of all possible `fetch` options with their default values (alternatives in comments): + +```js +let promise = fetch(url, { + method: "GET", // POST, PUT, DELETE, etc. + headers: { + // the content type header value is usually auto-set + // depending on the request body + "Content-Type": "text/plain;charset=UTF-8" + }, + body: undefined, // string, FormData, Blob, BufferSource, or URLSearchParams + referrer: "about:client", // or "" to send no Referer header, + // or an url from the current origin + referrerPolicy: "strict-origin-when-cross-origin", // no-referrer-when-downgrade, no-referrer, origin, same-origin... + mode: "cors", // same-origin, no-cors + credentials: "same-origin", // omit, include + cache: "default", // no-store, reload, no-cache, force-cache, or only-if-cached + redirect: "follow", // manual, error + integrity: "", // a hash, like "sha256-abcdef1234567890" + keepalive: false, // true + signal: undefined, // AbortController to abort request + window: window // null +}); +``` + +An impressive list, right? + +We fully covered `method`, `headers` and `body` in the chapter <info:fetch>. + +The `signal` option is covered in <info:fetch-abort>. + +Now let's explore the remaining capabilities. + +## referrer, referrerPolicy + +These options govern how `fetch` sets the HTTP `Referer` header. + +Usually that header is set automatically and contains the url of the page that made the request. In most scenarios, it's not important at all, sometimes, for security purposes, it makes sense to remove or shorten it. + +**The `referrer` option allows to set any `Referer` (within the current origin) or remove it.** + +To send no referrer, set an empty string: +```js +fetch('/page', { +*!* + referrer: "" // no Referer header +*/!* +}); +``` + +To set another url within the current origin: + +```js +fetch('/page', { + // assuming we're on https://javascript.info + // we can set any Referer header, but only within the current origin +*!* + referrer: "https://javascript.info/anotherpage" +*/!* +}); +``` + +**The `referrerPolicy` option sets general rules for `Referer`.** + +Requests are split into 3 types: + +1. Request to the same origin. +2. Request to another origin. +3. Request from HTTPS to HTTP (from safe to unsafe protocol). + +Unlike the `referrer` option that allows to set the exact `Referer` value, `referrerPolicy` tells the browser general rules for each request type. + +Possible values are described in the [Referrer Policy specification](https://w3c.github.io/webappsec-referrer-policy/): + +- **`"strict-origin-when-cross-origin"`** -- the default value: for same-origin send the full `Referer`, for cross-origin send only the origin, unless it's HTTPS→HTTP request, then send nothing. +- **`"no-referrer-when-downgrade"`** -- full `Referer` is always sent, unless we send a request from HTTPS to HTTP (to the less secure protocol). +- **`"no-referrer"`** -- never send `Referer`. +- **`"origin"`** -- only send the origin in `Referer`, not the full page URL, e.g. only `http://site.com` instead of `http://site.com/path`. +- **`"origin-when-cross-origin"`** -- send the full `Referer` to the same origin, but only the origin part for cross-origin requests (as above). +- **`"same-origin"`** -- send the full `Referer` to the same origin, but no `Referer` for cross-origin requests. +- **`"strict-origin"`** -- send only the origin, not the `Referer` for HTTPS→HTTP requests. +- **`"unsafe-url"`** -- always send the full url in `Referer`, even for HTTPS→HTTP requests. + +Here's a table with all combinations: + +| Value | To same origin | To another origin | HTTPS→HTTP | +|-------|----------------|-------------------|------------| +| `"no-referrer"` | - | - | - | +| `"no-referrer-when-downgrade"` | full | full | - | +| `"origin"` | origin | origin | origin | +| `"origin-when-cross-origin"` | full | origin | origin | +| `"same-origin"` | full | - | - | +| `"strict-origin"` | origin | origin | - | +| `"strict-origin-when-cross-origin"` or `""` (default) | full | origin | - | +| `"unsafe-url"` | full | full | full | + +Let's say we have an admin zone with a URL structure that shouldn't be known from outside of the site. + +If we send a `fetch`, then by default it always sends the `Referer` header with the full url of our page (except when we request from HTTPS to HTTP, then no `Referer`). + +E.g. `Referer: https://javascript.info/admin/secret/paths`. + +If we'd like other websites know only the origin part, not the URL-path, we can set the option: + +```js +fetch('https://another.com/page', { + // ... + referrerPolicy: "origin-when-cross-origin" // Referer: https://javascript.info +}); +``` + +We can put it to all `fetch` calls, maybe integrate into JavaScript library of our project that does all requests and uses `fetch` inside. + +Its only difference compared to the default behavior is that for requests to another origin `fetch` sends only the origin part of the URL (e.g. `https://javascript.info`, without path). For requests to our origin we still get the full `Referer` (maybe useful for debugging purposes). + +```smart header="Referrer policy is not only for `fetch`" +Referrer policy, described in the [specification](https://w3c.github.io/webappsec-referrer-policy/), is not just for `fetch`, but more global. + +In particular, it's possible to set the default policy for the whole page using the `Referrer-Policy` HTTP header, or per-link, with `<a rel="noreferrer">`. +``` + +## mode + +The `mode` option is a safe-guard that prevents occasional cross-origin requests: + +- **`"cors"`** -- the default, cross-origin requests are allowed, as described in <info:fetch-crossorigin>, +- **`"same-origin"`** -- cross-origin requests are forbidden, +- **`"no-cors"`** -- only safe cross-origin requests are allowed. + +This option may be useful when the URL for `fetch` comes from a 3rd-party, and we want a "power off switch" to limit cross-origin capabilities. + +## credentials + +The `credentials` option specifies whether `fetch` should send cookies and HTTP-Authorization headers with the request. + +- **`"same-origin"`** -- the default, don't send for cross-origin requests, +- **`"include"`** -- always send, requires `Access-Control-Allow-Credentials` from cross-origin server in order for JavaScript to access the response, that was covered in the chapter <info:fetch-crossorigin>, +- **`"omit"`** -- never send, even for same-origin requests. + +## cache + +By default, `fetch` requests make use of standard HTTP-caching. That is, it respects the `Expires` and `Cache-Control` headers, sends `If-Modified-Since` and so on. Just like regular HTTP-requests do. + +The `cache` options allows to ignore HTTP-cache or fine-tune its usage: + +- **`"default"`** -- `fetch` uses standard HTTP-cache rules and headers, +- **`"no-store"`** -- totally ignore HTTP-cache, this mode becomes the default if we set a header `If-Modified-Since`, `If-None-Match`, `If-Unmodified-Since`, `If-Match`, or `If-Range`, +- **`"reload"`** -- don't take the result from HTTP-cache (if any), but populate the cache with the response (if the response headers permit this action), +- **`"no-cache"`** -- create a conditional request if there is a cached response, and a normal request otherwise. Populate HTTP-cache with the response, +- **`"force-cache"`** -- use a response from HTTP-cache, even if it's stale. If there's no response in HTTP-cache, make a regular HTTP-request, behave normally, +- **`"only-if-cached"`** -- use a response from HTTP-cache, even if it's stale. If there's no response in HTTP-cache, then error. Only works when `mode` is `"same-origin"`. + +## redirect + +Normally, `fetch` transparently follows HTTP-redirects, like 301, 302 etc. + +The `redirect` option allows to change that: + +- **`"follow"`** -- the default, follow HTTP-redirects, +- **`"error"`** -- error in case of HTTP-redirect, +- **`"manual"`** -- allows to process HTTP-redirects manually. In case of redirect, we'll get a special response object, with `response.type="opaqueredirect"` and zeroed/empty status and most other properies. + +## integrity + +The `integrity` option allows to check if the response matches the known-ahead checksum. + +As described in the [specification](https://w3c.github.io/webappsec-subresource-integrity/), supported hash-functions are SHA-256, SHA-384, and SHA-512, there might be others depending on the browser. + +For example, we're downloading a file, and we know that its SHA-256 checksum is "abcdef" (a real checksum is longer, of course). + +We can put it in the `integrity` option, like this: + +```js +fetch('http://site.com/file', { + integrity: 'sha256-abcdef' +}); +``` + +Then `fetch` will calculate SHA-256 on its own and compare it with our string. In case of a mismatch, an error is triggered. + +## keepalive + +The `keepalive` option indicates that the request may "outlive" the webpage that initiated it. + +For example, we gather statistics on how the current visitor uses our page (mouse clicks, page fragments he views), to analyze and improve the user experience. + +When the visitor leaves our page -- we'd like to save the data to our server. + +We can use the `window.onunload` event for that: + +```js run +window.onunload = function() { + fetch('/analytics', { + method: 'POST', + body: "statistics", +*!* + keepalive: true +*/!* + }); +}; +``` + +Normally, when a document is unloaded, all associated network requests are aborted. But the `keepalive` option tells the browser to perform the request in the background, even after it leaves the page. So this option is essential for our request to succeed. + +It has a few limitations: + +- We can't send megabytes: the body limit for `keepalive` requests is 64KB. + - If we need to gather a lot of statistics about the visit, we should send it out regularly in packets, so that there won't be a lot left for the last `onunload` request. + - This limit applies to all `keepalive` requests together. In other words, we can perform multiple `keepalive` requests in parallel, but the sum of their body lengths should not exceed 64KB. +- We can't handle the server response if the document is unloaded. So in our example `fetch` will succeed due to `keepalive`, but subsequent functions won't work. + - In most cases, such as sending out statistics, it's not a problem, as the server just accepts the data and usually sends an empty response to such requests. diff --git a/5-network/05-fetch-api/logo-fetch.svg b/5-network/06-fetch-api/logo-fetch.svg similarity index 100% rename from 5-network/05-fetch-api/logo-fetch.svg rename to 5-network/06-fetch-api/logo-fetch.svg diff --git a/5-network/05-fetch-api/post.view/index.html b/5-network/06-fetch-api/post.view/index.html similarity index 100% rename from 5-network/05-fetch-api/post.view/index.html rename to 5-network/06-fetch-api/post.view/index.html diff --git a/5-network/05-fetch-api/post.view/server.js b/5-network/06-fetch-api/post.view/server.js similarity index 100% rename from 5-network/05-fetch-api/post.view/server.js rename to 5-network/06-fetch-api/post.view/server.js diff --git a/5-network/07-url/article.md b/5-network/07-url/article.md new file mode 100644 index 000000000..58b3ab1ae --- /dev/null +++ b/5-network/07-url/article.md @@ -0,0 +1,217 @@ + +# URL objects + +The built-in [URL](https://url.spec.whatwg.org/#api) class provides a convenient interface for creating and parsing URLs. + +There are no networking methods that require exactly a `URL` object, strings are good enough. So technically we don't have to use `URL`. But sometimes it can be really helpful. + +## Creating a URL + +The syntax to create a new `URL` object: + +```js +new URL(url, [base]) +``` + +- **`url`** -- the full URL or only path (if base is set, see below), +- **`base`** -- an optional base URL: if set and `url` argument has only path, then the URL is generated relative to `base`. + +For example: + +```js +let url = new URL('https://javascript.info/profile/admin'); +``` + +These two URLs are same: + +```js run +let url1 = new URL('https://javascript.info/profile/admin'); +let url2 = new URL('/profile/admin', 'https://javascript.info'); + +alert(url1); // https://javascript.info/profile/admin +alert(url2); // https://javascript.info/profile/admin +``` + +We can easily create a new URL based on the path relative to an existing URL: + +```js run +let url = new URL('https://javascript.info/profile/admin'); +let newUrl = new URL('tester', url); + +alert(newUrl); // https://javascript.info/profile/tester +``` + +The `URL` object immediately allows us to access its components, so it's a nice way to parse the url, e.g.: + +```js run +let url = new URL('https://javascript.info/url'); + +alert(url.protocol); // https: +alert(url.host); // javascript.info +alert(url.pathname); // /url +``` + +Here's the cheatsheet for URL components: + + + +- `href` is the full url, same as `url.toString()` +- `protocol` ends with the colon character `:` +- `search` - a string of parameters, starts with the question mark `?` +- `hash` starts with the hash character `#` +- there may be also `user` and `password` properties if HTTP authentication is present: `http://login:password@site.com` (not painted above, rarely used). + + +```smart header="We can pass `URL` objects to networking (and most other) methods instead of a string" +We can use a `URL` object in `fetch` or `XMLHttpRequest`, almost everywhere where a URL-string is expected. + +Generally, the `URL` object can be passed to any method instead of a string, as most methods will perform the string conversion, that turns a `URL` object into a string with full URL. +``` + +## SearchParams "?..." + +Let's say we want to create a url with given search params, for instance, `https://google.com/search?query=JavaScript`. + +We can provide them in the URL string: + +```js +new URL('https://google.com/search?query=JavaScript') +``` + +...But parameters need to be encoded if they contain spaces, non-latin letters, etc (more about that below). + +So there's a URL property for that: `url.searchParams`, an object of type [URLSearchParams](https://url.spec.whatwg.org/#urlsearchparams). + +It provides convenient methods for search parameters: + +- **`append(name, value)`** -- add the parameter by `name`, +- **`delete(name)`** -- remove the parameter by `name`, +- **`get(name)`** -- get the parameter by `name`, +- **`getAll(name)`** -- get all parameters with the same `name` (that's possible, e.g. `?user=John&user=Pete`), +- **`has(name)`** -- check for the existence of the parameter by `name`, +- **`set(name, value)`** -- set/replace the parameter, +- **`sort()`** -- sort parameters by name, rarely needed, +- ...and it's also iterable, similar to `Map`. + +An example with parameters that contain spaces and punctuation marks: + +```js run +let url = new URL('https://google.com/search'); + +url.searchParams.set('q', 'test me!'); // added parameter with a space and ! + +alert(url); // https://google.com/search?q=test+me%21 + +url.searchParams.set('tbs', 'qdr:y'); // added parameter with a colon : + +// parameters are automatically encoded +alert(url); // https://google.com/search?q=test+me%21&tbs=qdr%3Ay + +// iterate over search parameters (decoded) +for(let [name, value] of url.searchParams) { + alert(`${name}=${value}`); // q=test me!, then tbs=qdr:y +} +``` + + +## Encoding + +There's a standard [RFC3986](https://tools.ietf.org/html/rfc3986) that defines which characters are allowed in URLs and which are not. + +Those that are not allowed, must be encoded, for instance non-latin letters and spaces - replaced with their UTF-8 codes, prefixed by `%`, such as `%20` (a space can be encoded by `+`, for historical reasons, but that's an exception). + +The good news is that `URL` objects handle all that automatically. We just supply all parameters unencoded, and then convert the `URL` to string: + +```js run +// using some cyrillic characters for this example + +let url = new URL('https://ru.wikipedia.org/wiki/Тест'); + +url.searchParams.set('key', 'ъ'); +alert(url); //https://ru.wikipedia.org/wiki/%D0%A2%D0%B5%D1%81%D1%82?key=%D1%8A +``` + +As you can see, both `Тест` in the url path and `ъ` in the parameter are encoded. + +The URL became longer, because each cyrillic letter is represented with two bytes in UTF-8, so there are two `%..` entities. + +### Encoding strings + +In old times, before `URL` objects appeared, people used strings for URLs. + +As of now, `URL` objects are often more convenient, but strings can still be used as well. In many cases using a string makes the code shorter. + +If we use a string though, we need to encode/decode special characters manually. + +There are built-in functions for that: + +- [encodeURI](mdn:/JavaScript/Reference/Global_Objects/encodeURI) - encodes URL as a whole. +- [decodeURI](mdn:/JavaScript/Reference/Global_Objects/decodeURI) - decodes it back. +- [encodeURIComponent](mdn:/JavaScript/Reference/Global_Objects/encodeURIComponent) - encodes a URL component, such as a search parameter, or a hash, or a pathname. +- [decodeURIComponent](mdn:/JavaScript/Reference/Global_Objects/decodeURIComponent) - decodes it back. + +A natural question is: "What's the difference between `encodeURIComponent` and `encodeURI`? When we should use either?" + +That's easy to understand if we look at the URL, that's split into components in the picture above: + +``` +https://site.com:8080/path/page?p1=v1&p2=v2#hash +``` + +As we can see, characters such as `:`, `?`, `=`, `&`, `#` are allowed in URL. + +...On the other hand, if we look at a single URL component, such as a search parameter, these characters must be encoded, not to break the formatting. + +- `encodeURI` encodes only characters that are totally forbidden in URL. +- `encodeURIComponent` encodes same characters, and, in addition to them, characters `#`, `$`, `&`, `+`, `,`, `/`, `:`, `;`, `=`, `?` and `@`. + +So, for a whole URL we can use `encodeURI`: + +```js run +// using cyrillic characters in url path +let url = encodeURI('http://site.com/привет'); + +alert(url); // http://site.com/%D0%BF%D1%80%D0%B8%D0%B2%D0%B5%D1%82 +``` + +...While for URL parameters we should use `encodeURIComponent` instead: + +```js run +let music = encodeURIComponent('Rock&Roll'); + +let url = `https://google.com/search?q=${music}`; +alert(url); // https://google.com/search?q=Rock%26Roll +``` + +Compare it with `encodeURI`: + +```js run +let music = encodeURI('Rock&Roll'); + +let url = `https://google.com/search?q=${music}`; +alert(url); // https://google.com/search?q=Rock&Roll +``` + +As we can see, `encodeURI` does not encode `&`, as this is a legit character in URL as a whole. + +But we should encode `&` inside a search parameter, otherwise, we get `q=Rock&Roll` - that is actually `q=Rock` plus some obscure parameter `Roll`. Not as intended. + +So we should use only `encodeURIComponent` for each search parameter, to correctly insert it in the URL string. The safest is to encode both name and value, unless we're absolutely sure that it has only allowed characters. + +````smart header="Encoding difference compared to `URL`" +Classes [URL](https://url.spec.whatwg.org/#url-class) and [URLSearchParams](https://url.spec.whatwg.org/#interface-urlsearchparams) are based on the latest URI specification: [RFC3986](https://tools.ietf.org/html/rfc3986), while `encode*` functions are based on the obsolete version [RFC2396](https://www.ietf.org/rfc/rfc2396.txt). + +There are a few differences, e.g. IPv6 addresses are encoded differently: + +```js run +// valid url with IPv6 address +let url = 'http://[2607:f8b0:4005:802::1007]/'; + +alert(encodeURI(url)); // http://%5B2607:f8b0:4005:802::1007%5D/ +alert(new URL(url)); // http://[2607:f8b0:4005:802::1007]/ +``` + +As we can see, `encodeURI` replaced square brackets `[...]`, that's not correct, the reason is: IPv6 urls did not exist at the time of RFC2396 (August 1998). + +Such cases are rare, `encode*` functions work well most of the time. +```` diff --git a/5-network/07-url/url-object.svg b/5-network/07-url/url-object.svg new file mode 100644 index 000000000..628ccc13b --- /dev/null +++ b/5-network/07-url/url-object.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="698" height="246" viewBox="0 0 698 246"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="url-object.svg"><path id="Line-Copy" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M27 38v180"/><path id="Line-Copy-2" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M109 132v86"/><path id="Line-Copy-6" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M136 112v106"/><path id="Line-Copy-10" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M27 38l665 .5"/><path id="Line-Copy-11" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M27 74.5h280"/><path id="Line-Copy-7" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M254 132v86"/><path id="Line-Copy-8" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M243 132v86"/><path id="Line-Copy-9" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M308 38v180"/><path id="Line-Copy-13" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M439 132v86"/><path id="Line-Copy-14" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M622 132v86"/><path id="Line-Copy-15" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M692 39v179"/><text id="href" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="337" y="33">href</tspan></text><text id="origin" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="135" y="68">origin</tspan></text><path id="Line-Copy-12" fill="#C06334" stroke="#C06334" stroke-dasharray="1" stroke-linecap="square" stroke-width="2" d="M136 112.5h170"/><text id="host" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="202" y="106">host</tspan></text><text id="protocol" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="30.6" y="152">protocol</tspan></text><path id="https://site.com:8080/path/page?p1=v1&p2=v2…#hash" fill="#181717" fill-rule="nonzero" d="M31.772 196.4h2.808v4.716h.09a4.03 4.03 0 011.296-.981c.492-.234 1.104-.351 1.836-.351.576 0 1.077.06 1.503.18.426.12.774.33 1.044.63s.471.711.603 1.233c.132.522.198 1.185.198 1.989V209h-1.404v-4.914c0-.516-.03-.966-.09-1.35-.06-.384-.177-.705-.351-.963a1.626 1.626 0 00-.711-.585c-.3-.132-.684-.198-1.152-.198a2.76 2.76 0 00-.954.171c-.312.114-.597.27-.855.468s-.48.438-.666.72a2.693 2.693 0 00-.387.927V209h-1.386v-11.394h-1.422V196.4zm13.686 3.6H47.6v-1.782l1.404-.396V200h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611v-4.464h-2.142V200zm13.2 0H60.8v-1.782l1.404-.396V200h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611v-4.464h-2.142V200zm14.37 3.15a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V200h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm15.216 5.472c0-.384-.15-.681-.45-.891-.3-.21-.672-.378-1.116-.504a20.643 20.643 0 00-1.449-.351 8.36 8.36 0 01-1.449-.432 3.247 3.247 0 01-1.116-.729c-.3-.306-.45-.735-.45-1.287 0-.456.099-.846.297-1.17.198-.324.459-.591.783-.801.324-.21.702-.366 1.134-.468a5.856 5.856 0 011.35-.153c.84 0 1.563.105 2.169.315.606.21 1.089.429 1.449.657l-.576 1.134a15.16 15.16 0 00-1.287-.621c-.462-.198-1.041-.297-1.737-.297-.264 0-.525.027-.783.081a2.767 2.767 0 00-.702.243c-.21.108-.378.249-.504.423a1.06 1.06 0 00-.189.639c0 .312.15.552.45.72.3.168.672.309 1.116.423.444.114.927.222 1.449.324a7.044 7.044 0 011.449.441c.444.192.816.45 1.116.774.3.324.45.762.45 1.314 0 .828-.327 1.512-.981 2.052-.654.54-1.635.81-2.943.81-.396 0-.786-.036-1.17-.108a7.096 7.096 0 01-1.08-.288 6.498 6.498 0 01-.918-.405 3.86 3.86 0 01-.684-.459l.72-1.17c.144.144.336.291.576.441a5.437 5.437 0 001.719.693 4.833 4.833 0 001.908.027c.282-.054.531-.138.747-.252.216-.114.387-.264.513-.45s.189-.411.189-.675zm9.51-5.634c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm17.124-11.826l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm13.2 0l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm12.966 10.386c0-.384-.15-.681-.45-.891-.3-.21-.672-.378-1.116-.504a20.643 20.643 0 00-1.449-.351 8.36 8.36 0 01-1.449-.432 3.247 3.247 0 01-1.116-.729c-.3-.306-.45-.735-.45-1.287 0-.456.099-.846.297-1.17.198-.324.459-.591.783-.801.324-.21.702-.366 1.134-.468a5.856 5.856 0 011.35-.153c.84 0 1.563.105 2.169.315.606.21 1.089.429 1.449.657l-.576 1.134a15.16 15.16 0 00-1.287-.621c-.462-.198-1.041-.297-1.737-.297-.264 0-.525.027-.783.081a2.767 2.767 0 00-.702.243c-.21.108-.378.249-.504.423a1.06 1.06 0 00-.189.639c0 .312.15.552.45.72.3.168.672.309 1.116.423.444.114.927.222 1.449.324a7.044 7.044 0 011.449.441c.444.192.816.45 1.116.774.3.324.45.762.45 1.314 0 .828-.327 1.512-.981 2.052-.654.54-1.635.81-2.943.81-.396 0-.786-.036-1.17-.108a7.096 7.096 0 01-1.08-.288 6.498 6.498 0 01-.918-.405 3.86 3.86 0 01-.684-.459l.72-1.17c.144.144.336.291.576.441a5.437 5.437 0 001.719.693 4.833 4.833 0 001.908.027c.282-.054.531-.138.747-.252.216-.114.387-.264.513-.45s.189-.411.189-.675zm7.206 2.43v-1.206h3.132v-6.588h-3.132V200h4.572v7.794h3.06V209h-7.632zm2.646-11.556c0-.324.108-.603.324-.837.216-.234.492-.351.828-.351.348 0 .639.117.873.351.234.234.351.513.351.837 0 .312-.117.576-.351.792a1.238 1.238 0 01-.873.324c-.336 0-.612-.108-.828-.324a1.077 1.077 0 01-.324-.792zm9.222 2.556h2.142v-1.782l1.404-.396V200h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611v-4.464h-2.142V200zm22.164 7.866a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm11.67 7.038c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm16.908-6.624a7.111 7.111 0 00-.891-.243 4.64 4.64 0 00-.945-.099c-1.176 0-2.052.276-2.628.828-.576.552-.864 1.428-.864 2.628 0 .528.084.999.252 1.413.168.414.408.765.72 1.053.312.288.687.51 1.125.666.438.156.921.234 1.449.234.564 0 1.113-.096 1.647-.288.534-.192.981-.444 1.341-.756l.63 1.044a5.095 5.095 0 01-.63.45 6.053 6.053 0 01-1.971.774 6.38 6.38 0 01-1.305.126c-.78 0-1.467-.111-2.061-.333a3.914 3.914 0 01-1.485-.954 4.083 4.083 0 01-.9-1.494 5.818 5.818 0 01-.306-1.935c0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.888 0 1.611.078 2.169.234.558.156 1.029.33 1.413.522l-.018.054v2.502h-1.296v-1.71zm6.324 3.114c0-1.452.378-2.601 1.134-3.447.756-.846 1.836-1.269 3.24-1.269.756 0 1.407.123 1.953.369s.999.579 1.359.999c.36.42.627.918.801 1.494.174.576.261 1.194.261 1.854 0 .72-.096 1.371-.288 1.953a4.057 4.057 0 01-.846 1.485 3.738 3.738 0 01-1.377.945c-.546.222-1.167.333-1.863.333-.744 0-1.392-.123-1.944-.369a3.815 3.815 0 01-1.368-.999 4.073 4.073 0 01-.801-1.494 6.381 6.381 0 01-.261-1.854zm1.494 0c0 .42.051.84.153 1.26.102.42.267.798.495 1.134.228.336.525.606.891.81.366.204.813.306 1.341.306.96 0 1.683-.297 2.169-.891.486-.594.729-1.467.729-2.619 0-.432-.051-.855-.153-1.269a3.353 3.353 0 00-.504-1.125 2.656 2.656 0 00-.9-.81c-.366-.204-.813-.306-1.341-.306-.96 0-1.68.294-2.16.882-.48.588-.72 1.464-.72 2.628zm15.396 4.5v-5.994a9.51 9.51 0 00-.027-.729 2.442 2.442 0 00-.126-.63 1.01 1.01 0 00-.288-.441.734.734 0 00-.495-.162c-.408 0-.753.168-1.035.504-.282.336-.495.756-.639 1.26V209h-1.368v-9h.936l.27 1.098h.072c.12-.18.237-.351.351-.513.114-.162.246-.303.396-.423.15-.12.327-.213.531-.279.204-.066.462-.099.774-.099.18 0 .366.027.558.081.192.054.369.138.531.252.162.114.303.267.423.459s.198.426.234.702c.276-.468.585-.834.927-1.098.342-.264.813-.396 1.413-.396.396 0 .717.066.963.198s.438.321.576.567c.138.246.234.54.288.882.054.342.081.723.081 1.143V209h-1.368v-6.12c0-.252-.012-.489-.036-.711a2.268 2.268 0 00-.135-.585.967.967 0 00-.27-.396c-.114-.096-.267-.144-.459-.144-.42 0-.774.168-1.062.504-.288.336-.504.816-.648 1.44V209h-1.368zm12.678-8.064c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm0 7.074c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm10.356-2.07c0-1.452.882-2.622 2.646-3.51-.336-.18-.648-.366-.936-.558a3.746 3.746 0 01-.747-.648 2.737 2.737 0 01-.495-.828 2.996 2.996 0 01-.18-1.08c0-.444.09-.858.27-1.242.18-.384.438-.714.774-.99a3.842 3.842 0 011.224-.657 4.988 4.988 0 011.602-.243c.552 0 1.053.072 1.503.216a3.5 3.5 0 011.152.603c.318.258.561.561.729.909.168.348.252.72.252 1.116 0 .684-.171 1.311-.513 1.881-.342.57-.891 1.089-1.647 1.557.348.18.672.372.972.576.3.204.561.432.783.684.222.252.396.537.522.855.126.318.189.681.189 1.089 0 .504-.09.975-.27 1.413a3 3 0 01-.801 1.125 3.88 3.88 0 01-1.305.738c-.516.18-1.116.27-1.8.27-.636 0-1.2-.09-1.692-.27a3.696 3.696 0 01-1.233-.72c-.33-.3-.579-.648-.747-1.044a3.148 3.148 0 01-.252-1.242zm6.678-.18c0-.348-.081-.657-.243-.927a2.74 2.74 0 00-.648-.729 5.59 5.59 0 00-.927-.594c-.348-.18-.708-.36-1.08-.54-.864.42-1.479.882-1.845 1.386-.366.504-.549.978-.549 1.422 0 .3.057.585.171.855.114.27.282.507.504.711.222.204.498.366.828.486.33.12.711.18 1.143.18.348 0 .681-.048.999-.144.318-.096.6-.237.846-.423s.441-.42.585-.702c.144-.282.216-.609.216-.981zm-4.986-6.498c0 .336.078.636.234.9.156.264.363.501.621.711.258.21.555.402.891.576.336.174.69.345 1.062.513.636-.42 1.113-.84 1.431-1.26.318-.42.477-.882.477-1.386 0-.3-.063-.567-.189-.801a1.948 1.948 0 00-.504-.603 2.292 2.292 0 00-.729-.387 2.77 2.77 0 00-.864-.135c-.384 0-.726.054-1.026.162-.3.108-.552.252-.756.432-.204.18-.363.381-.477.603a1.463 1.463 0 00-.171.675zm11.148 3.438c0-1.02.09-1.935.27-2.745.18-.81.45-1.494.81-2.052.36-.558.816-.984 1.368-1.278.552-.294 1.206-.441 1.962-.441.804 0 1.485.144 2.043.432a3.466 3.466 0 011.359 1.26c.348.552.6 1.233.756 2.043.156.81.234 1.737.234 2.781 0 1.02-.09 1.935-.27 2.745-.18.81-.45 1.494-.81 2.052-.36.558-.816.984-1.368 1.278-.552.294-1.206.441-1.962.441-.792 0-1.467-.159-2.025-.477a3.8 3.8 0 01-1.368-1.341c-.354-.576-.609-1.263-.765-2.061a13.774 13.774 0 01-.234-2.637zm7.326 0c0-.636-.036-1.242-.108-1.818l-5.346 4.878c.204.684.51 1.23.918 1.638.408.408.942.612 1.602.612 1.056 0 1.809-.438 2.259-1.314.45-.876.675-2.208.675-3.996zm-5.832 0c0 .3.009.588.027.864.018.276.039.546.063.81l5.364-4.86c-.204-.648-.507-1.164-.909-1.548-.402-.384-.945-.576-1.629-.576-1.068 0-1.821.441-2.259 1.323-.438.882-.657 2.211-.657 3.987zm12.066 3.24c0-1.452.882-2.622 2.646-3.51-.336-.18-.648-.366-.936-.558a3.746 3.746 0 01-.747-.648 2.737 2.737 0 01-.495-.828 2.996 2.996 0 01-.18-1.08c0-.444.09-.858.27-1.242.18-.384.438-.714.774-.99a3.842 3.842 0 011.224-.657 4.988 4.988 0 011.602-.243c.552 0 1.053.072 1.503.216a3.5 3.5 0 011.152.603c.318.258.561.561.729.909.168.348.252.72.252 1.116 0 .684-.171 1.311-.513 1.881-.342.57-.891 1.089-1.647 1.557.348.18.672.372.972.576.3.204.561.432.783.684.222.252.396.537.522.855.126.318.189.681.189 1.089 0 .504-.09.975-.27 1.413a3 3 0 01-.801 1.125 3.88 3.88 0 01-1.305.738c-.516.18-1.116.27-1.8.27-.636 0-1.2-.09-1.692-.27a3.696 3.696 0 01-1.233-.72c-.33-.3-.579-.648-.747-1.044a3.148 3.148 0 01-.252-1.242zm6.678-.18c0-.348-.081-.657-.243-.927a2.74 2.74 0 00-.648-.729 5.59 5.59 0 00-.927-.594c-.348-.18-.708-.36-1.08-.54-.864.42-1.479.882-1.845 1.386-.366.504-.549.978-.549 1.422 0 .3.057.585.171.855.114.27.282.507.504.711.222.204.498.366.828.486.33.12.711.18 1.143.18.348 0 .681-.048.999-.144.318-.096.6-.237.846-.423s.441-.42.585-.702c.144-.282.216-.609.216-.981zm-4.986-6.498c0 .336.078.636.234.9.156.264.363.501.621.711.258.21.555.402.891.576.336.174.69.345 1.062.513.636-.42 1.113-.84 1.431-1.26.318-.42.477-.882.477-1.386 0-.3-.063-.567-.189-.801a1.948 1.948 0 00-.504-.603 2.292 2.292 0 00-.729-.387 2.77 2.77 0 00-.864-.135c-.384 0-.726.054-1.026.162-.3.108-.552.252-.756.432-.204.18-.363.381-.477.603a1.463 1.463 0 00-.171.675zm11.148 3.438c0-1.02.09-1.935.27-2.745.18-.81.45-1.494.81-2.052.36-.558.816-.984 1.368-1.278.552-.294 1.206-.441 1.962-.441.804 0 1.485.144 2.043.432a3.466 3.466 0 011.359 1.26c.348.552.6 1.233.756 2.043.156.81.234 1.737.234 2.781 0 1.02-.09 1.935-.27 2.745-.18.81-.45 1.494-.81 2.052-.36.558-.816.984-1.368 1.278-.552.294-1.206.441-1.962.441-.792 0-1.467-.159-2.025-.477a3.8 3.8 0 01-1.368-1.341c-.354-.576-.609-1.263-.765-2.061a13.774 13.774 0 01-.234-2.637zm7.326 0c0-.636-.036-1.242-.108-1.818l-5.346 4.878c.204.684.51 1.23.918 1.638.408.408.942.612 1.602.612 1.056 0 1.809-.438 2.259-1.314.45-.876.675-2.208.675-3.996zm-5.832 0c0 .3.009.588.027.864.018.276.039.546.063.81l5.364-4.86c-.204-.648-.507-1.164-.909-1.548-.402-.384-.945-.576-1.629-.576-1.068 0-1.821.441-2.259 1.323-.438.882-.657 2.211-.657 3.987zm18.834-6.516l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm6.81 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V200h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm9.276-.396a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V209h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm9.168-7.956h2.142v-1.782l1.404-.396V200h4.806v1.206h-4.806v4.248c0 .876.213 1.524.639 1.944.426.42 1.035.63 1.827.63.54 0 1.011-.102 1.413-.306a7.614 7.614 0 001.089-.666l.468 1.062c-.42.336-.921.606-1.503.81a5.377 5.377 0 01-1.791.306c-.48 0-.933-.069-1.359-.207a3.12 3.12 0 01-1.125-.639 3.06 3.06 0 01-.774-1.107c-.192-.45-.288-.987-.288-1.611v-4.464h-2.142V200zm12.714-3.6h2.808v4.716h.09a4.03 4.03 0 011.296-.981c.492-.234 1.104-.351 1.836-.351.576 0 1.077.06 1.503.18.426.12.774.33 1.044.63s.471.711.603 1.233c.132.522.198 1.185.198 1.989V209h-1.404v-4.914c0-.516-.03-.966-.09-1.35-.06-.384-.177-.705-.351-.963a1.626 1.626 0 00-.711-.585c-.3-.132-.684-.198-1.152-.198a2.76 2.76 0 00-.954.171c-.312.114-.597.27-.855.468s-.48.438-.666.72a2.693 2.693 0 00-.387.927V209h-1.386v-11.394h-1.422V196.4zm21.246-.216l1.134.504-6.57 14.832-1.134-.504 6.57-14.832zm6.81 6.966a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V200h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm9.276-.396a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V209h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm18.042 1.494c0 .564-.108 1.059-.324 1.485a2.917 2.917 0 01-.891 1.053 4.02 4.02 0 01-1.323.621 6.185 6.185 0 01-1.638.207c-.708 0-1.326-.072-1.854-.216a5.515 5.515 0 01-1.368-.558l.684-1.278c.108.096.249.189.423.279.174.09.375.174.603.252.228.078.471.141.729.189.258.048.513.072.765.072.528 0 .969-.048 1.323-.144.354-.096.639-.246.855-.45.216-.204.372-.468.468-.792.096-.324.144-.72.144-1.188v-1.008h-.072c-.276.396-.63.702-1.062.918-.432.216-.99.324-1.674.324-1.368 0-2.373-.381-3.015-1.143-.642-.762-.963-1.947-.963-3.555 0-1.536.408-2.697 1.224-3.483.816-.786 2.028-1.179 3.636-1.179.744 0 1.377.051 1.899.153.522.102.999.231 1.431.387v9.054zm-4.014-1.44c.732 0 1.308-.189 1.728-.567.42-.378.714-.957.882-1.737v-4.266c-.528-.252-1.266-.378-2.214-.378-.96 0-1.716.282-2.268.846-.552.564-.828 1.428-.828 2.592 0 .516.048.987.144 1.413.096.426.249.795.459 1.107.21.312.486.555.828.729.342.174.765.261 1.269.261zm17.304-.144a4.53 4.53 0 01-.765.54 5.645 5.645 0 01-.963.432 7.06 7.06 0 01-1.089.279 6.633 6.633 0 01-1.143.099c-.72 0-1.356-.111-1.908-.333a3.674 3.674 0 01-1.386-.954 4.158 4.158 0 01-.846-1.485c-.192-.576-.288-1.224-.288-1.944 0-.756.105-1.428.315-2.016.21-.588.513-1.08.909-1.476a3.959 3.959 0 011.44-.909c.564-.21 1.194-.315 1.89-.315.504 0 1.002.066 1.494.198a2.95 2.95 0 011.305.747c.378.366.669.87.873 1.512.204.642.276 1.473.216 2.493h-6.966c0 1.08.291 1.887.873 2.421.582.534 1.359.801 2.331.801.324 0 .645-.039.963-.117.318-.078.621-.171.909-.279.288-.108.543-.228.765-.36.222-.132.393-.252.513-.36l.558 1.026zm-3.798-6.894a4.86 4.86 0 00-1.125.126 2.564 2.564 0 00-.936.423c-.27.198-.492.456-.666.774-.174.318-.285.711-.333 1.179h5.598c-.06-.792-.315-1.407-.765-1.845-.45-.438-1.041-.657-1.773-.657zm11.652 4.842a1.011 1.011 0 01-.018-.198v-.198c0-.504.093-.936.279-1.296.186-.36.417-.681.693-.963a6.74 6.74 0 01.891-.765c.318-.228.615-.471.891-.729s.507-.543.693-.855c.186-.312.279-.684.279-1.116 0-.228-.054-.474-.162-.738a2.314 2.314 0 00-.495-.738 2.501 2.501 0 00-.864-.558c-.354-.144-.777-.216-1.269-.216a3.7 3.7 0 00-1.071.144 4.343 4.343 0 00-.864.36 4.644 4.644 0 00-.702.477c-.21.174-.399.339-.567.495l-.918-.828a4.62 4.62 0 011.953-1.431c.798-.318 1.713-.477 2.745-.477.624 0 1.17.102 1.638.306.468.204.855.465 1.161.783.306.318.534.672.684 1.062.15.39.225.771.225 1.143 0 .576-.099 1.059-.297 1.449-.198.39-.447.738-.747 1.044-.3.306-.621.582-.963.828-.342.246-.663.51-.963.792-.3.282-.549.597-.747.945-.198.348-.297.774-.297 1.278h-1.188zm-.36 2.358c0-.324.093-.579.279-.765.186-.186.435-.279.747-.279.324 0 .582.093.774.279.192.186.288.441.288.765 0 .312-.096.564-.288.756-.192.192-.45.288-.774.288-.312 0-.561-.096-.747-.288-.186-.192-.279-.444-.279-.756zm11.112-5.022a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V200h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.561 6.561 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm9.618 6.606h2.826v-9.468l-2.88 2.016-.702-1.026 3.96-2.826h1.008v11.304h2.772V209h-6.984v-1.296zm12.138-6.93h8.352v1.296h-8.352v-1.296zm0 2.952h8.352v1.296h-8.352v-1.296zm17.412 3.528h.198L502.16 200h1.566l-3.798 9h-1.584l-3.87-9h1.656l3.006 7.254zm10.05.45h2.826v-9.468l-2.88 2.016-.702-1.026 3.96-2.826h1.008v11.304h2.772V209h-6.984v-1.296zm11.454-2.43c0-.792.222-1.548.666-2.268.444-.72 1.062-1.326 1.854-1.818a11.046 11.046 0 01-.747-1.296c-.21-.432-.315-.9-.315-1.404 0-.288.048-.57.144-.846.096-.276.249-.522.459-.738.21-.216.48-.39.81-.522.33-.132.729-.198 1.197-.198.492 0 .906.063 1.242.189.336.126.606.291.81.495.204.204.351.432.441.684.09.252.135.51.135.774 0 .504-.195 1.023-.585 1.557-.39.534-1.005 1.047-1.845 1.539.192.348.417.705.675 1.071.258.366.531.729.819 1.089.288.36.582.717.882 1.071.3.354.6.681.9.981a3.19 3.19 0 00.351-.702c.102-.276.186-.564.252-.864a8.368 8.368 0 00.189-1.728h1.278a10.485 10.485 0 01-.36 2.259c-.204.75-.504 1.383-.9 1.899a5.657 5.657 0 001.35 1.188v1.53c-.636-.312-1.362-.87-2.178-1.674-.216.216-.435.423-.657.621a4.088 4.088 0 01-1.692.909 4.734 4.734 0 01-1.233.144c-.492 0-.972-.081-1.44-.243a3.787 3.787 0 01-1.26-.729 3.608 3.608 0 01-.9-1.224c-.228-.492-.342-1.074-.342-1.746zm6.678 1.368c-.348-.348-.69-.726-1.026-1.134a34.295 34.295 0 01-1.809-2.403 30.16 30.16 0 01-.657-.999c-.648.492-1.101.99-1.359 1.494a3.505 3.505 0 00-.387 1.62c0 .468.087.873.261 1.215.174.342.393.627.657.855.264.228.555.396.873.504.318.108.627.162.927.162.6 0 1.101-.135 1.503-.405.402-.27.741-.573 1.017-.909zm-3.816-8.154c0 .36.075.717.225 1.071.15.354.345.705.585 1.053.636-.408 1.077-.78 1.323-1.116.246-.336.369-.63.369-.882 0-.384-.093-.696-.279-.936-.186-.24-.507-.36-.963-.36-.432 0-.75.108-.954.324-.204.216-.306.498-.306.846zm11.526 4.662a18.063 18.063 0 00-.081-1.512 5.988 5.988 0 00-.063-.468h-1.206V200h2.394l.18 1.26h.09a4.06 4.06 0 011.233-1.053c.498-.282 1.107-.423 1.827-.423 1.284 0 2.25.354 2.898 1.062.648.708.972 1.866.972 3.474 0 .756-.111 1.437-.333 2.043a4.295 4.295 0 01-.945 1.539c-.408.42-.9.744-1.476.972a5.141 5.141 0 01-1.908.342c-.252 0-.477-.009-.675-.027a5.93 5.93 0 01-.54-.072 4.255 4.255 0 01-.477-.117 6.562 6.562 0 01-.504-.18v3.78h-1.386v-9.45zm4.14-2.052c-.36 0-.699.066-1.017.198a3.034 3.034 0 00-1.449 1.242 2.118 2.118 0 00-.288.792v4.104a3.3 3.3 0 00.891.423c.33.102.765.153 1.305.153.96 0 1.728-.315 2.304-.945.576-.63.864-1.551.864-2.763 0-1.02-.201-1.809-.603-2.367-.402-.558-1.071-.837-2.007-.837zm16.332-1.512c0 1.104-.522 2.334-1.566 3.69-1.044 1.356-2.502 2.832-4.374 4.428h6.336V209h-7.992v-1.296c.228-.216.54-.498.936-.846a57.05 57.05 0 001.278-1.161c.456-.426.921-.885 1.395-1.377a15.64 15.64 0 001.287-1.512 8.867 8.867 0 00.936-1.548c.24-.516.36-1.014.36-1.494 0-.72-.189-1.287-.567-1.701-.378-.414-.945-.621-1.701-.621-.648 0-1.194.072-1.638.216a3.988 3.988 0 00-1.206.63l-.612-.99a5.806 5.806 0 011.719-.846 6.962 6.962 0 011.989-.27c1.128 0 1.98.306 2.556.918.576.612.864 1.44.864 2.484zm5.424 1.188h8.352v1.296h-8.352v-1.296zm0 2.952h8.352v1.296h-8.352v-1.296zm17.412 3.528h.198L581.36 200h1.566l-3.798 9h-1.584l-3.87-9h1.656l3.006 7.254zm16.764-7.668c0 1.104-.522 2.334-1.566 3.69-1.044 1.356-2.502 2.832-4.374 4.428h6.336V209h-7.992v-1.296c.228-.216.54-.498.936-.846s.822-.735 1.278-1.161c.456-.426.921-.885 1.395-1.377a15.64 15.64 0 001.287-1.512 8.867 8.867 0 00.936-1.548c.24-.516.36-1.014.36-1.494 0-.72-.189-1.287-.567-1.701-.378-.414-.945-.621-1.701-.621-.648 0-1.194.072-1.638.216a3.988 3.988 0 00-1.206.63l-.612-.99a5.806 5.806 0 011.719-.846 6.962 6.962 0 011.989-.27c1.128 0 1.98.306 2.556.918.576.612.864 1.44.864 2.484zm12.966 8.424c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm-4.572 0c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm-4.536 0c0-.384.108-.681.324-.891.216-.21.504-.315.864-.315.384 0 .684.105.9.315.216.21.324.507.324.891 0 .348-.108.636-.324.864-.216.228-.516.342-.9.342-.36 0-.648-.114-.864-.342-.216-.228-.324-.516-.324-.864zm32.898-3.06h-2.61l-.702 3.006h-1.242l.702-3.006h-1.764l.252-1.152h1.782l.558-2.358h-1.692l.252-1.152h1.71l.684-2.844h1.242l-.684 2.844h2.61l.684-2.844h1.242l-.684 2.844h1.746l-.288 1.152h-1.728l-.558 2.358h1.674l-.288 1.152h-1.656l-.702 3.006h-1.242l.702-3.006zm-2.34-1.152h2.61l.558-2.358h-2.61l-.558 2.358zm9.456-7.398h2.808v4.716h.09a4.03 4.03 0 011.296-.981c.492-.234 1.104-.351 1.836-.351.576 0 1.077.06 1.503.18.426.12.774.33 1.044.63s.471.711.603 1.233c.132.522.198 1.185.198 1.989V209h-1.404v-4.914c0-.516-.03-.966-.09-1.35-.06-.384-.177-.705-.351-.963a1.626 1.626 0 00-.711-.585c-.3-.132-.684-.198-1.152-.198a2.76 2.76 0 00-.954.171c-.312.114-.597.27-.855.468s-.48.438-.666.72a2.693 2.693 0 00-.387.927V209h-1.386v-11.394h-1.422V196.4zm15.072 4.302a5.717 5.717 0 011.746-.648 9.544 9.544 0 011.908-.198c.612 0 1.113.096 1.503.288.39.192.696.441.918.747.222.306.372.654.45 1.044.078.39.117.783.117 1.179 0 .456-.012.942-.036 1.458a66.688 66.688 0 00-.054 1.548c0 .6.036 1.17.108 1.71h1.206V209h-2.394l-.162-1.35h-.09c-.072.108-.18.246-.324.414a2.964 2.964 0 01-.567.495 3.664 3.664 0 01-.855.423c-.336.12-.732.18-1.188.18-.888 0-1.59-.228-2.106-.684-.516-.456-.774-1.08-.774-1.872 0-.612.135-1.122.405-1.53.27-.408.657-.72 1.161-.936.504-.216 1.113-.342 1.827-.378.714-.036 1.515.012 2.403.144.06-.552.069-1.011.027-1.377-.042-.366-.138-.657-.288-.873a1.206 1.206 0 00-.63-.459c-.27-.09-.603-.135-.999-.135-.54 0-1.056.075-1.548.225-.492.15-.93.303-1.314.459l-.45-1.044zm2.646 7.254c.336 0 .648-.054.936-.162.288-.108.54-.246.756-.414a2.743 2.743 0 00.864-1.116v-1.26c-.624-.108-1.2-.162-1.728-.162s-.984.057-1.368.171c-.384.114-.684.291-.9.531-.216.24-.324.552-.324.936 0 .396.135.741.405 1.035.27.294.723.441 1.359.441zm16.494-1.386c0-.384-.15-.681-.45-.891-.3-.21-.672-.378-1.116-.504a20.643 20.643 0 00-1.449-.351 8.36 8.36 0 01-1.449-.432 3.247 3.247 0 01-1.116-.729c-.3-.306-.45-.735-.45-1.287 0-.456.099-.846.297-1.17.198-.324.459-.591.783-.801.324-.21.702-.366 1.134-.468a5.856 5.856 0 011.35-.153c.84 0 1.563.105 2.169.315.606.21 1.089.429 1.449.657l-.576 1.134a15.16 15.16 0 00-1.287-.621c-.462-.198-1.041-.297-1.737-.297-.264 0-.525.027-.783.081a2.767 2.767 0 00-.702.243c-.21.108-.378.249-.504.423a1.06 1.06 0 00-.189.639c0 .312.15.552.45.72.3.168.672.309 1.116.423.444.114.927.222 1.449.324a7.044 7.044 0 011.449.441c.444.192.816.45 1.116.774.3.324.45.762.45 1.314 0 .828-.327 1.512-.981 2.052-.654.54-1.635.81-2.943.81-.396 0-.786-.036-1.17-.108a7.096 7.096 0 01-1.08-.288 6.498 6.498 0 01-.918-.405 3.86 3.86 0 01-.684-.459l.72-1.17c.144.144.336.291.576.441a5.437 5.437 0 001.719.693 4.833 4.833 0 001.908.027c.282-.054.531-.138.747-.252.216-.114.387-.264.513-.45s.189-.411.189-.675zm5.388-10.17h2.808v4.716h.09a4.03 4.03 0 011.296-.981c.492-.234 1.104-.351 1.836-.351.576 0 1.077.06 1.503.18.426.12.774.33 1.044.63s.471.711.603 1.233c.132.522.198 1.185.198 1.989V209h-1.404v-4.914c0-.516-.03-.966-.09-1.35-.06-.384-.177-.705-.351-.963a1.626 1.626 0 00-.711-.585c-.3-.132-.684-.198-1.152-.198a2.76 2.76 0 00-.954.171c-.312.114-.597.27-.855.468s-.48.438-.666.72a2.693 2.693 0 00-.387.927V209h-1.386v-11.394h-1.422V196.4z"/><text id="hostname" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="147" y="152">hostname</tspan></text><text id="port" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="261" y="152">port</tspan></text><text id="pathname" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="334" y="152">pathname</tspan></text><text id="search" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="496" y="152">search</tspan></text><text id="hash" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="16" font-weight="normal"><tspan x="637" y="152">hash</tspan></text></g></g></svg> \ No newline at end of file diff --git a/5-network/07-xmlhttprequest/article.md b/5-network/08-xmlhttprequest/article.md similarity index 78% rename from 5-network/07-xmlhttprequest/article.md rename to 5-network/08-xmlhttprequest/article.md index bda4bd1b7..43d816cab 100644 --- a/5-network/07-xmlhttprequest/article.md +++ b/5-network/08-xmlhttprequest/article.md @@ -2,19 +2,19 @@ `XMLHttpRequest` is a built-in browser object that allows to make HTTP requests in JavaScript. -Despite of having the word "XML" in its name, it can operate on any data, not only in XML format. We can upload/download files, track progress and much more. +Despite having the word "XML" in its name, it can operate on any data, not only in XML format. We can upload/download files, track progress and much more. Right now, there's another, more modern method `fetch`, that somewhat deprecates `XMLHttpRequest`. -In modern web-development `XMLHttpRequest` may be used for three reasons: +In modern web-development `XMLHttpRequest` is used for three reasons: 1. Historical reasons: we need to support existing scripts with `XMLHttpRequest`. 2. We need to support old browsers, and don't want polyfills (e.g. to keep scripts tiny). 3. We need something that `fetch` can't do yet, e.g. to track upload progress. -Does that sound familiar? If yes, then all right, go on with `XMLHttpRequest`. Otherwise, please head on to fetch (coming soon). +Does that sound familiar? If yes, then all right, go on with `XMLHttpRequest`. Otherwise, please head on to <info:fetch>. -## Basic flow +## The basics XMLHttpRequest has two modes of operation: synchronous and asynchronous. @@ -22,20 +22,21 @@ Let's see the asynchronous first, as it's used in the majority of cases. To do the request, we need 3 steps: -1. Create `XMLHttpRequest`. +1. Create `XMLHttpRequest`: ```js - let xhr = new XMLHttpRequest(); // no arguments + let xhr = new XMLHttpRequest(); ``` + The constructor has no arguments. -2. Initialize it. +2. Initialize it, usually right after `new XMLHttpRequest`: ```js xhr.open(method, URL, [async, user, password]) ``` - This method is usually called first after `new XMLHttpRequest`. It specifies the main parameters of the request: + This method specifies the main parameters of the request: - `method` -- HTTP-method. Usually `"GET"` or `"POST"`. - - `URL` -- the URL to request. + - `URL` -- the URL to request, a string, can be [URL](info:url) object. - `async` -- if explicitly set to `false`, then the request is synchronous, we'll cover that a bit later. - `user`, `password` -- login and password for basic HTTP auth (if required). @@ -49,14 +50,14 @@ To do the request, we need 3 steps: This method opens the connection and sends the request to server. The optional `body` parameter contains the request body. - Some request methods like `GET` do not have a body. And some of them like `POST` use `body` to send the data to the server. We'll see examples later. + Some request methods like `GET` do not have a body. And some of them like `POST` use `body` to send the data to the server. We'll see examples of that later. -4. Listen to events for response. +4. Listen to `xhr` events for response. - These three are the most widely used: - - `load` -- when the result is ready, that includes HTTP errors like 404. + These three events are the most widely used: + - `load` -- when the request is complete (even if HTTP status is like 400 or 500), and the response is fully downloaded. - `error` -- when the request couldn't be made, e.g. network down or invalid URL. - - `progress` -- triggers periodically during the download, reports how much downloaded. + - `progress` -- triggers periodically while the response is being downloaded, reports how much has been downloaded. ```js xhr.onload = function() { @@ -92,7 +93,7 @@ xhr.onload = function() { if (xhr.status != 200) { // analyze HTTP status of the response alert(`Error ${xhr.status}: ${xhr.statusText}`); // e.g. 404: Not Found } else { // show the result - alert(`Done, got ${xhr.response.length} bytes`); // responseText is the server + alert(`Done, got ${xhr.response.length} bytes`); // response is the server response } }; @@ -110,7 +111,7 @@ xhr.onerror = function() { }; ``` -Once the server has responded, we can receive the result in the following properties of the request object: +Once the server has responded, we can receive the result in the following `xhr` properties: `status` : HTTP status code (a number): `200`, `404`, `403` and so on, can be `0` in case of a non-HTTP failure. @@ -119,23 +120,28 @@ Once the server has responded, we can receive the result in the following proper : HTTP status message (a string): usually `OK` for `200`, `Not Found` for `404`, `Forbidden` for `403` and so on. `response` (old scripts may use `responseText`) -: The server response. +: The server response body. -If we changed our mind, we can terminate the request at any time. The call to `xhr.abort()` does that: +We can also specify a timeout using the corresponding property: ```js -xhr.abort(); // terminate the request +xhr.timeout = 10000; // timeout in ms, 10 seconds ``` -That triggers `abort` event. +If the request does not succeed within the given time, it gets canceled and `timeout` event triggers. -We can also specify a timeout using the corresponding property: +````smart header="URL search parameters" +To add parameters to URL, like `?name=value`, and ensure the proper encoding, we can use [URL](info:url) object: ```js -xhr.timeout = 10000; // timeout in ms, 10 seconds +let url = new URL('https://google.com/search'); +url.searchParams.set('q', 'test me!'); + +// the parameter 'q' is encoded +xhr.open('GET', url); // https://google.com/search?q=test+me%21 ``` -If the request does not succeed within the given time, it gets canceled and `timeout` event triggers. +```` ## Response Type @@ -143,9 +149,9 @@ We can use `xhr.responseType` property to set the response format: - `""` (default) -- get as string, - `"text"` -- get as string, -- `"arraybuffer"` -- get as `ArrayBuffer` (for binary data, see chapter <info:arraybuffer-and-views>), +- `"arraybuffer"` -- get as `ArrayBuffer` (for binary data, see chapter <info:arraybuffer-binary-arrays>), - `"blob"` -- get as `Blob` (for binary data, see chapter <info:blob>), -- `"document"` -- get as XML document (can use XPath and other XML methods), +- `"document"` -- get as XML document (can use XPath and other XML methods) or HTML document (based on the MIME type of the received data), - `"json"` -- get as JSON (parsed automatically). For example, let's get the response as JSON: @@ -184,7 +190,7 @@ All states, as in [the specification](https://xhr.spec.whatwg.org/#states): UNSENT = 0; // initial state OPENED = 1; // open called HEADERS_RECEIVED = 2; // response headers received -LOADING = 3; // response is loading (a data packed is received) +LOADING = 3; // response is loading (a data packet is received) DONE = 4; // request complete ``` @@ -203,9 +209,17 @@ xhr.onreadystatechange = function() { }; ``` -You can find `readystatechange` listeners in really old code, for historical reasons. +You can find `readystatechange` listeners in really old code, it's there for historical reasons, as there was a time when there were no `load` and other events. Nowadays, `load/error/progress` handlers deprecate it. + +## Aborting request + +We can terminate the request at any time. The call to `xhr.abort()` does that: + +```js +xhr.abort(); // terminate the request +``` -Nowadays, `load/error/progress` handlers deprecate it. +That triggers `abort` event, and `xhr.status` becomes `0`. ## Synchronous requests @@ -229,7 +243,7 @@ try { } } catch(err) { // instead of onerror alert("Request failed"); -}; +} ``` It might look good, but synchronous calls are used rarely, because they block in-page JavaScript till the loading is complete. In some browsers it becomes impossible to scroll. If a synchronous call takes too much time, the browser may suggest to close the "hanging" webpage. @@ -255,13 +269,13 @@ There are 3 methods for HTTP-headers: ```warn header="Headers limitations" Several headers are managed exclusively by the browser, e.g. `Referer` and `Host`. - The full list is [in the specification](http://www.w3.org/TR/XMLHttpRequest/#the-setrequestheader-method). + The full list is [in the specification](https://xhr.spec.whatwg.org/#the-setrequestheader()-method). - XMLHttpRequest is not allowed to change them, for the sake of user safety and correctness of the request. + `XMLHttpRequest` is not allowed to change them, for the sake of user safety and correctness of the request. ``` ````warn header="Can't remove a header" - Another peciliarity of `XMLHttpRequest` is that one can't undo `setRequestHeader`. + Another peculiarity of `XMLHttpRequest` is that one can't undo `setRequestHeader`. Once the header is set, it's set. Additional calls add information to the header, don't overwrite it. @@ -290,7 +304,7 @@ There are 3 methods for HTTP-headers: Headers are returned as a single line, e.g.: - ``` + ```http Cache-Control: max-age=31536000 Content-Length: 4260 Content-Type: image/png @@ -312,11 +326,13 @@ There are 3 methods for HTTP-headers: result[name] = value; return result; }, {}); + + // headers['Content-Type'] = 'image/png' ``` ## POST, FormData -To make a POST request, we can use the built-in [FormData](https://developer.mozilla.org/en-US/docs/Web/API/FormData) object. +To make a POST request, we can use the built-in [FormData](mdn:api/FormData) object. The syntax: @@ -325,14 +341,14 @@ let formData = new FormData([form]); // creates an object, optionally fill from formData.append(name, value); // appends a field ``` -Create it, optionally from a form, `append` more fields if needed, and then: +We create it, optionally fill from a form, `append` more fields if needed, and then: 1. `xhr.open('POST', ...)` – use `POST` method. 2. `xhr.send(formData)` to submit the form to the server. For instance: -```html run +```html run refresh <form name="person"> <input name="name" value="John"> <input name="surname" value="Smith"> @@ -350,6 +366,7 @@ For instance: xhr.open("POST", "/article/xmlhttprequest/post/user"); xhr.send(formData); + xhr.onload = () => alert(xhr.response); </script> ``` @@ -373,19 +390,19 @@ xhr.setRequestHeader('Content-type', 'application/json; charset=utf-8'); xhr.send(json); ``` -The `.send(body)` method is pretty omnivore. It can send almost everything, including Blob and BufferSource objects. +The `.send(body)` method is pretty omnivore. It can send almost any `body`, including `Blob` and `BufferSource` objects. ## Upload progress -The `progress` event only works on the downloading stage. +The `progress` event triggers only on the downloading stage. -That is: if we `POST` something, `XMLHttpRequest` first uploads our data, then downloads the response. +That is: if we `POST` something, `XMLHttpRequest` first uploads our data (the request body), then downloads the response. -If we're uploading something big, then we're surely more interested in tracking the upload progress. But `progress` event doesn't help here. +If we're uploading something big, then we're surely more interested in tracking the upload progress. But `xhr.onprogress` doesn't help here. -There's another object `xhr.upload`, without methods, exclusively for upload events. +There's another object, without methods, exclusively to track upload events: `xhr.upload`. -Here's the list: +It generates events, similar to `xhr`, but `xhr.upload` triggers them solely on uploading: - `loadstart` -- upload started. - `progress` -- triggers periodically during the upload. @@ -444,7 +461,7 @@ function upload(file) { ## Cross-origin requests -`XMLHttpRequest` can make cross-domain requests, using the same CORS policy as [fetch](info:fetch-crossorigin). +`XMLHttpRequest` can make cross-origin requests, using the same CORS policy as [fetch](info:fetch-crossorigin). Just like `fetch`, it doesn't send cookies and HTTP-authorization to another origin by default. To enable them, set `xhr.withCredentials` to `true`: @@ -458,6 +475,8 @@ xhr.open('POST', 'http://anywhere.com/request'); ... ``` +See the chapter <info:fetch-crossorigin> for details about cross-origin headers. + ## Summary @@ -468,7 +487,7 @@ let xhr = new XMLHttpRequest(); xhr.open('GET', '/my/url'); -xhr.send(); // for POST, can send a string or FormData +xhr.send(); xhr.onload = function() { if (xhr.status != 200) { // HTTP error? @@ -490,17 +509,19 @@ xhr.onerror = function() { }; ``` -There are actually more events, the [modern specification](http://www.w3.org/TR/XMLHttpRequest/#events) lists them (in the lifecycle order): +There are actually more events, the [modern specification](https://xhr.spec.whatwg.org/#events) lists them (in the lifecycle order): - `loadstart` -- the request has started. -- `progress` -- a data packet of the response has arrived, the whole response body at the moment is in `responseText`. +- `progress` -- a data packet of the response has arrived, the whole response body at the moment is in `response`. - `abort` -- the request was canceled by the call `xhr.abort()`. -- `error` -- connection error has occured, e.g. wrong domain name. Doesn't happen for HTTP-errors like 404. +- `error` -- connection error has occurred, e.g. wrong domain name. Doesn't happen for HTTP-errors like 404. - `load` -- the request has finished successfully. - `timeout` -- the request was canceled due to timeout (only happens if it was set). -- `loadend` -- the request has finished (succeffully or not). +- `loadend` -- triggers after `load`, `error`, `timeout` or `abort`. + +The `error`, `abort`, `timeout`, and `load` events are mutually exclusive. Only one of them may happen. -The most used events are load completion (`load`), load failure (`error`), and also `progress` to track the progress. +The most used events are load completion (`load`), load failure (`error`), or we can use a single `loadend` handler and check the properties of the request object `xhr` to see what happened. We've already seen another event: `readystatechange`. Historically, it appeared long ago, before the specification settled. Nowadays, there's no need to use it, we can replace it with newer events, but it can often be found in older scripts. diff --git a/5-network/07-xmlhttprequest/example.view/index.html b/5-network/08-xmlhttprequest/example.view/index.html similarity index 100% rename from 5-network/07-xmlhttprequest/example.view/index.html rename to 5-network/08-xmlhttprequest/example.view/index.html diff --git a/5-network/07-xmlhttprequest/example.view/server.js b/5-network/08-xmlhttprequest/example.view/server.js similarity index 100% rename from 5-network/07-xmlhttprequest/example.view/server.js rename to 5-network/08-xmlhttprequest/example.view/server.js diff --git a/5-network/07-xmlhttprequest/hello.txt b/5-network/08-xmlhttprequest/hello.txt similarity index 100% rename from 5-network/07-xmlhttprequest/hello.txt rename to 5-network/08-xmlhttprequest/hello.txt diff --git a/5-network/07-xmlhttprequest/phones-async.view/index.html b/5-network/08-xmlhttprequest/phones-async.view/index.html similarity index 100% rename from 5-network/07-xmlhttprequest/phones-async.view/index.html rename to 5-network/08-xmlhttprequest/phones-async.view/index.html diff --git a/5-network/07-xmlhttprequest/phones-async.view/phones.json b/5-network/08-xmlhttprequest/phones-async.view/phones.json similarity index 100% rename from 5-network/07-xmlhttprequest/phones-async.view/phones.json rename to 5-network/08-xmlhttprequest/phones-async.view/phones.json diff --git a/5-network/07-xmlhttprequest/phones-async.view/server.js b/5-network/08-xmlhttprequest/phones-async.view/server.js similarity index 100% rename from 5-network/07-xmlhttprequest/phones-async.view/server.js rename to 5-network/08-xmlhttprequest/phones-async.view/server.js diff --git a/5-network/07-xmlhttprequest/phones.json b/5-network/08-xmlhttprequest/phones.json similarity index 100% rename from 5-network/07-xmlhttprequest/phones.json rename to 5-network/08-xmlhttprequest/phones.json diff --git a/5-network/07-xmlhttprequest/phones.view/index.html b/5-network/08-xmlhttprequest/phones.view/index.html similarity index 100% rename from 5-network/07-xmlhttprequest/phones.view/index.html rename to 5-network/08-xmlhttprequest/phones.view/index.html diff --git a/5-network/07-xmlhttprequest/phones.view/phones.json b/5-network/08-xmlhttprequest/phones.view/phones.json similarity index 100% rename from 5-network/07-xmlhttprequest/phones.view/phones.json rename to 5-network/08-xmlhttprequest/phones.view/phones.json diff --git a/5-network/07-xmlhttprequest/phones.view/server.js b/5-network/08-xmlhttprequest/phones.view/server.js similarity index 100% rename from 5-network/07-xmlhttprequest/phones.view/server.js rename to 5-network/08-xmlhttprequest/phones.view/server.js diff --git a/5-network/07-xmlhttprequest/post.view/index.html b/5-network/08-xmlhttprequest/post.view/index.html similarity index 100% rename from 5-network/07-xmlhttprequest/post.view/index.html rename to 5-network/08-xmlhttprequest/post.view/index.html diff --git a/5-network/07-xmlhttprequest/post.view/server.js b/5-network/08-xmlhttprequest/post.view/server.js similarity index 100% rename from 5-network/07-xmlhttprequest/post.view/server.js rename to 5-network/08-xmlhttprequest/post.view/server.js diff --git a/5-network/09-resume-upload/article.md b/5-network/09-resume-upload/article.md new file mode 100644 index 000000000..b0aa447d6 --- /dev/null +++ b/5-network/09-resume-upload/article.md @@ -0,0 +1,82 @@ +# Resumable file upload + +With `fetch` method it's fairly easy to upload a file. + +How to resume the upload after lost connection? There's no built-in option for that, but we have the pieces to implement it. + +Resumable uploads should come with upload progress indication, as we expect big files (if we may need to resume). So, as `fetch` doesn't allow to track upload progress, we'll use [XMLHttpRequest](info:xmlhttprequest). + +## Not-so-useful progress event + +To resume upload, we need to know how much was uploaded till the connection was lost. + +There's `xhr.upload.onprogress` to track upload progress. + +Unfortunately, it won't help us to resume the upload here, as it triggers when the data is *sent*, but was it received by the server? The browser doesn't know. + +Maybe it was buffered by a local network proxy, or maybe the remote server process just died and couldn't process them, or it was just lost in the middle and didn't reach the receiver. + +That's why this event is only useful to show a nice progress bar. + +To resume upload, we need to know *exactly* the number of bytes received by the server. And only the server can tell that, so we'll make an additional request. + +## Algorithm + +1. First, create a file id, to uniquely identify the file we're going to upload: + ```js + let fileId = file.name + '-' + file.size + '-' + file.lastModified; + ``` + That's needed for resume upload, to tell the server what we're resuming. + + If the name or the size or the last modification date changes, then there'll be another `fileId`. + +2. Send a request to the server, asking how many bytes it already has, like this: + ```js + let response = await fetch('status', { + headers: { + 'X-File-Id': fileId + } + }); + + // The server has that many bytes + let startByte = +await response.text(); + ``` + + This assumes that the server tracks file uploads by `X-File-Id` header. Should be implemented at server-side. + + If the file doesn't yet exist at the server, then the server response should be `0` + +3. Then, we can use `Blob` method `slice` to send the file from `startByte`: + ```js + xhr.open("POST", "upload"); + + // File id, so that the server knows which file we upload + xhr.setRequestHeader('X-File-Id', fileId); + + // The byte we're resuming from, so the server knows we're resuming + xhr.setRequestHeader('X-Start-Byte', startByte); + + xhr.upload.onprogress = (e) => { + console.log(`Uploaded ${startByte + e.loaded} of ${startByte + e.total}`); + }; + + // file can be from input.files[0] or another source + xhr.send(file.slice(startByte)); + ``` + + Here we send the server both file id as `X-File-Id`, so it knows which file we're uploading, and the starting byte as `X-Start-Byte`, so it knows we're not uploading it initially, but resuming. + + The server should check its records, and if there was an upload of that file, and the current uploaded size is exactly `X-Start-Byte`, then append the data to it. + + +Here's the demo with both client and server code, written on Node.js. + +It works only partially on this site, as Node.js is behind another server named Nginx, that buffers uploads, passing them to Node.js when fully complete. + +But you can download it and run locally for the full demonstration: + +[codetabs src="upload-resume" height=200] + +As we can see, modern networking methods are close to file managers in their capabilities -- control over headers, progress indicator, sending file parts, etc. + +We can implement resumable upload and much more. diff --git a/5-network/09-resume-upload/upload-resume.view/index.html b/5-network/09-resume-upload/upload-resume.view/index.html new file mode 100644 index 000000000..f1178145e --- /dev/null +++ b/5-network/09-resume-upload/upload-resume.view/index.html @@ -0,0 +1,50 @@ +<!DOCTYPE HTML> + +<script src="uploader.js"></script> + +<form name="upload" method="POST" enctype="multipart/form-data" action="/upload"> + <input type="file" name="myfile"> + <input type="submit" name="submit" value="Upload (Resumes automatically)"> +</form> + +<button onclick="uploader.stop()">Stop upload</button> + + +<div id="log">Progress indication</div> + +<script> + function log(html) { + document.getElementById('log').innerHTML = html; + console.log(html); + } + + function onProgress(loaded, total) { + log("progress " + loaded + ' / ' + total); + } + + let uploader; + + document.forms.upload.onsubmit = async function(e) { + e.preventDefault(); + + let file = this.elements.myfile.files[0]; + if (!file) return; + + uploader = new Uploader({file, onProgress}); + + try { + let uploaded = await uploader.upload(); + + if (uploaded) { + log('success'); + } else { + log('stopped'); + } + + } catch(err) { + console.error(err); + log('error'); + } + }; + +</script> diff --git a/5-network/09-resume-upload/upload-resume.view/server.js b/5-network/09-resume-upload/upload-resume.view/server.js new file mode 100644 index 000000000..83ce59f7a --- /dev/null +++ b/5-network/09-resume-upload/upload-resume.view/server.js @@ -0,0 +1,123 @@ +let http = require('http'); +let static = require('node-static'); +let fileServer = new static.Server('.'); +let path = require('path'); +let fs = require('fs'); +let debug = require('debug')('example:resume-upload'); + +let uploads = Object.create(null); + +function onUpload(req, res) { + + let fileId = req.headers['x-file-id']; + let startByte = +req.headers['x-start-byte']; + + if (!fileId) { + res.writeHead(400, "No file id"); + res.end(); + } + + // we'll files "nowhere" + let filePath = '/dev/null'; + // could use a real path instead, e.g. + // let filePath = path.join('/tmp', fileId); + + debug("onUpload fileId: ", fileId); + + // initialize a new upload + if (!uploads[fileId]) uploads[fileId] = {}; + let upload = uploads[fileId]; + + debug("bytesReceived:" + upload.bytesReceived + " startByte:" + startByte) + + let fileStream; + + // if startByte is 0 or not set, create a new file, otherwise check the size and append to existing one + if (!startByte) { + upload.bytesReceived = 0; + fileStream = fs.createWriteStream(filePath, { + flags: 'w' + }); + debug("New file created: " + filePath); + } else { + // we can check on-disk file size as well to be sure + if (upload.bytesReceived != startByte) { + res.writeHead(400, "Wrong start byte"); + res.end(upload.bytesReceived); + return; + } + // append to existing file + fileStream = fs.createWriteStream(filePath, { + flags: 'a' + }); + debug("File reopened: " + filePath); + } + + + req.on('data', function(data) { + debug("bytes received", upload.bytesReceived); + upload.bytesReceived += data.length; + }); + + // send request body to file + req.pipe(fileStream); + + // when the request is finished, and all its data is written + fileStream.on('close', function() { + if (upload.bytesReceived == req.headers['x-file-size']) { + debug("Upload finished"); + delete uploads[fileId]; + + // can do something else with the uploaded file here + + res.end("Success " + upload.bytesReceived); + } else { + // connection lost, we leave the unfinished file around + debug("File unfinished, stopped at " + upload.bytesReceived); + res.end(); + } + }); + + // in case of I/O error - finish the request + fileStream.on('error', function(err) { + debug("fileStream error"); + res.writeHead(500, "File error"); + res.end(); + }); + +} + +function onStatus(req, res) { + let fileId = req.headers['x-file-id']; + let upload = uploads[fileId]; + debug("onStatus fileId:", fileId, " upload:", upload); + if (!upload) { + res.end("0") + } else { + res.end(String(upload.bytesReceived)); + } +} + + +function accept(req, res) { + if (req.url == '/status') { + onStatus(req, res); + } else if (req.url == '/upload' && req.method == 'POST') { + onUpload(req, res); + } else { + fileServer.serve(req, res); + } + +} + + + + +// ----------------------------------- + +if (!module.parent) { + http.createServer(accept).listen(8080); + console.log('Server listening at port 8080'); +} else { + exports.accept = accept; +} diff --git a/5-network/09-resume-upload/upload-resume.view/uploader.js b/5-network/09-resume-upload/upload-resume.view/uploader.js new file mode 100644 index 000000000..10002039c --- /dev/null +++ b/5-network/09-resume-upload/upload-resume.view/uploader.js @@ -0,0 +1,75 @@ +class Uploader { + + constructor({file, onProgress}) { + this.file = file; + this.onProgress = onProgress; + + // create fileId that uniquely identifies the file + // we could also add user session identifier (if had one), to make it even more unique + this.fileId = file.name + '-' + file.size + '-' + file.lastModified; + } + + async getUploadedBytes() { + let response = await fetch('status', { + headers: { + 'X-File-Id': this.fileId + } + }); + + if (response.status != 200) { + throw new Error("Can't get uploaded bytes: " + response.statusText); + } + + let text = await response.text(); + + return +text; + } + + async upload() { + this.startByte = await this.getUploadedBytes(); + + let xhr = this.xhr = new XMLHttpRequest(); + xhr.open("POST", "upload", true); + + // send file id, so that the server knows which file to resume + xhr.setRequestHeader('X-File-Id', this.fileId); + // send the byte we're resuming from, so the server knows we're resuming + xhr.setRequestHeader('X-Start-Byte', this.startByte); + + xhr.upload.onprogress = (e) => { + this.onProgress(this.startByte + e.loaded, this.startByte + e.total); + }; + + console.log("send the file, starting from", this.startByte); + xhr.send(this.file.slice(this.startByte)); + + // return + // true if upload was successful, + // false if aborted + // throw in case of an error + return await new Promise((resolve, reject) => { + + xhr.onload = xhr.onerror = () => { + console.log("upload end status:" + xhr.status + " text:" + xhr.statusText); + + if (xhr.status == 200) { + resolve(true); + } else { + reject(new Error("Upload failed: " + xhr.statusText)); + } + }; + + // onabort triggers only when xhr.abort() is called + xhr.onabort = () => resolve(false); + + }); + + } + + stop() { + if (this.xhr) { + this.xhr.abort(); + } + } + +} diff --git a/5-network/10-long-polling/article.md b/5-network/10-long-polling/article.md new file mode 100644 index 000000000..e9d8abe39 --- /dev/null +++ b/5-network/10-long-polling/article.md @@ -0,0 +1,98 @@ +# Long polling + +Long polling is the simplest way of having persistent connection with server, that doesn't use any specific protocol like WebSocket or Server Sent Events. + +Being very easy to implement, it's also good enough in a lot of cases. + +## Regular Polling + +The simplest way to get new information from the server is periodic polling. That is, regular requests to the server: "Hello, I'm here, do you have any information for me?". For example, once every 10 seconds. + +In response, the server first takes a notice to itself that the client is online, and second - sends a packet of messages it got till that moment. + +That works, but there are downsides: +1. Messages are passed with a delay up to 10 seconds (between requests). +2. Even if there are no messages, the server is bombed with requests every 10 seconds, even if the user switched somewhere else or is asleep. That's quite a load to handle, speaking performance-wise. + +So, if we're talking about a very small service, the approach may be viable, but generally, it needs an improvement. + +## Long polling + +So-called "long polling" is a much better way to poll the server. + +It's also very easy to implement, and delivers messages without delays. + +The flow: + +1. A request is sent to the server. +2. The server doesn't close the connection until it has a message to send. +3. When a message appears - the server responds to the request with it. +4. The browser makes a new request immediately. + +This situation, where the browser has sent a request and keeps a pending connection with the server, is standard for this method. Only when a message is delivered, the connection is closed and reestablished. + + + +If the connection is lost, because of, say, a network error, the browser immediately sends a new request. + +A sketch of client-side `subscribe` function that makes long requests: + +```js +async function subscribe() { + let response = await fetch("/subscribe"); + + if (response.status == 502) { + // Status 502 is a connection timeout error, + // may happen when the connection was pending for too long, + // and the remote server or a proxy closed it + // let's reconnect + await subscribe(); + } else if (response.status != 200) { + // An error - let's show it + showMessage(response.statusText); + // Reconnect in one second + await new Promise(resolve => setTimeout(resolve, 1000)); + await subscribe(); + } else { + // Get and show the message + let message = await response.text(); + showMessage(message); + // Call subscribe() again to get the next message + await subscribe(); + } +} + +subscribe(); +``` + +As you can see, `subscribe` function makes a fetch, then waits for the response, handles it and calls itself again. + +```warn header="Server should be ok with many pending connections" +The server architecture must be able to work with many pending connections. + +Certain server architectures run one process per connection, resulting in there being as many processes as there are connections, while each process consumes quite a bit of memory. So, too many connections will just consume it all. + +That's often the case for backends written in languages like PHP and Ruby. + +Servers written using Node.js usually don't have such problems. + +That said, it isn't a programming language issue. Most modern languages, including PHP and Ruby allow to implement a proper backend. Just please make sure that your server architecture works fine with many simultaneous connections. +``` + +## Demo: a chat + +Here's a demo chat, you can also download it and run locally (if you're familiar with Node.js and can install modules): + +[codetabs src="longpoll" height=500] + +Browser code is in `browser.js`. + +## Area of usage + +Long polling works great in situations when messages are rare. + +If messages come very often, then the chart of requesting-receiving messages, painted above, becomes saw-like. + +Every message is a separate request, supplied with headers, authentication overhead, and so on. + +So, in this case, another method is preferred, such as [Websocket](info:websocket) or [Server Sent Events](info:server-sent-events). diff --git a/5-network/10-long-polling/long-polling.svg b/5-network/10-long-polling/long-polling.svg new file mode 100644 index 000000000..045ef371f --- /dev/null +++ b/5-network/10-long-polling/long-polling.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="521" height="320" viewBox="0 0 521 320"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="long-polling.svg"><text id="Browser" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="27" y="74">Browser</tspan></text><text id="Server" fill="#181717" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal"><tspan x="31" y="226">Server</tspan></text><path id="Line" fill="#181717" fill-rule="nonzero" d="M450.81 75.82l.435.244 8 4.5.775.436-.775.436-8 4.5-.436.245-.49-.872.436-.245 6.336-3.564H66.5v-1h390.591l-6.336-3.564-.436-.245.49-.872zM450.81 200.32l.435.244 8 4.5.775.436-.775.436-8 4.5-.436.245-.49-.872.436-.245L457.09 206H65.5v-1h391.591l-6.336-3.564-.436-.245.49-.872z"/><text id="request" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold" transform="rotate(76 113.291 143.266)"><tspan x="83.891" y="146.266">request</tspan></text><text id="connection-hangs" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="126" y="227">connection</tspan> <tspan x="142.519" y="245">hangs</tspan></text><text id="connection-breaks-en" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="12" font-weight="normal"><tspan x="186.09" y="264">connection breaks</tspan> <tspan x="196.479" y="282">end of request</tspan></text><path id="Line-3" fill="#C06334" fill-rule="nonzero" d="M90.233 80.79l.238.972 27.169 110.901 5.828-1.427L120 206.5l-10.13-11.932 5.827-1.429L88.529 82.238l-.238-.971 1.942-.476z"/><text id="data" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold" transform="rotate(-76 206.38 143.266)"><tspan x="189.58" y="146.266">data</tspan></text><path id="Line-2" fill="#C06334" fill-rule="nonzero" d="M233.226 82l3.107 15.341-5.794-1.565-29.574 109.485-.26.965-1.931-.521.26-.966L228.61 95.255l-5.792-1.565L233.226 82z"/><path id="Line-2-Copy" fill="#C06334" fill-rule="nonzero" d="M377.226 82l3.107 15.341-5.794-1.565-29.574 109.485-.26.965-1.931-.521.26-.966L372.61 95.255l-5.792-1.565L377.226 82z"/><text id="request-copy" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold" transform="rotate(76 260.291 143.266)"><tspan x="230.891" y="146.266">request</tspan></text><text id="connection-hangs-copy" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="270" y="227">connection</tspan> <tspan x="286.519" y="245">hangs</tspan></text><path id="Line-3-Copy-3" fill="#C06334" fill-rule="nonzero" d="M237.233 80.79l.238.972 27.169 110.901 5.828-1.427L267 206.5l-10.13-11.932 5.827-1.429-27.168-110.901-.238-.971 1.942-.476z"/><text id="request-copy-2" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold" transform="rotate(76 404.291 143.266)"><tspan x="374.891" y="146.266">request</tspan></text><path id="Line-3-Copy-4" fill="#C06334" fill-rule="nonzero" d="M381.233 80.79l.238.972 27.169 110.901 5.828-1.427L411 206.5l-10.13-11.932 5.827-1.429-27.168-110.901-.238-.971 1.942-.476z"/><text id="data-copy" fill="#AF6E24" font-family="PTMono-Bold, PT Mono" font-size="14" font-weight="bold" transform="rotate(-76 350.38 143.266)"><tspan x="333.58" y="146.266">data</tspan></text><path id="Line" stroke="#7E7C7B" stroke-dasharray="3,3" stroke-linecap="square" d="M235 40.75v201"/><text id="connection-breaks-en-copy" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="12" font-weight="normal"><tspan x="330.09" y="264">connection breaks</tspan> <tspan x="340.479" y="282">end of request</tspan></text><path id="Line-Copy" stroke="#7E7C7B" stroke-dasharray="3,3" stroke-linecap="square" d="M379 40.75v201"/></g></g></svg> \ No newline at end of file diff --git a/5-network/10-long-polling/longpoll.view/browser.js b/5-network/10-long-polling/longpoll.view/browser.js new file mode 100644 index 000000000..3a66aa5c6 --- /dev/null +++ b/5-network/10-long-polling/longpoll.view/browser.js @@ -0,0 +1,54 @@ +// Sending messages, a simple POST +function PublishForm(form, url) { + + function sendMessage(message) { + fetch(url, { + method: 'POST', + body: message + }); + } + + form.onsubmit = function() { + let message = form.message.value; + if (message) { + form.message.value = ''; + sendMessage(message); + } + return false; + }; +} + +// Receiving messages with long polling +function SubscribePane(elem, url) { + + function showMessage(message) { + let messageElem = document.createElement('div'); + messageElem.append(message); + elem.append(messageElem); + } + + async function subscribe() { + let response = await fetch(url); + + if (response.status == 502) { + // Connection timeout + // happens when the connection was pending for too long + // let's reconnect + await subscribe(); + } else if (response.status != 200) { + // Show Error + showMessage(response.statusText); + // Reconnect in one second + await new Promise(resolve => setTimeout(resolve, 1000)); + await subscribe(); + } else { + // Got message + let message = await response.text(); + showMessage(message); + await subscribe(); + } + } + + subscribe(); + +} diff --git a/5-network/10-long-polling/longpoll.view/index.html b/5-network/10-long-polling/longpoll.view/index.html new file mode 100644 index 000000000..7452c1838 --- /dev/null +++ b/5-network/10-long-polling/longpoll.view/index.html @@ -0,0 +1,18 @@ +<!DOCTYPE html> +<script src="browser.js"></script> + +All visitors of this page will see messages of each other. + +<form name="publish"> + <input type="text" name="message" /> + <input type="submit" value="Send" /> +</form> + +<div id="subscribe"> +</div> + +<script> + new PublishForm(document.forms.publish, 'publish'); + // random url parameter to avoid any caching issues + new SubscribePane(document.getElementById('subscribe'), 'subscribe?random=' + Math.random()); +</script> diff --git a/5-network/10-long-polling/longpoll.view/server.js b/5-network/10-long-polling/longpoll.view/server.js new file mode 100644 index 000000000..c3903e375 --- /dev/null +++ b/5-network/10-long-polling/longpoll.view/server.js @@ -0,0 +1,87 @@ +let http = require('http'); +let url = require('url'); +let querystring = require('querystring'); +let static = require('node-static'); + +let fileServer = new static.Server('.'); + +let subscribers = Object.create(null); + +function onSubscribe(req, res) { + let id = Math.random(); + + res.setHeader('Content-Type', 'text/plain;charset=utf-8'); + res.setHeader("Cache-Control", "no-cache, must-revalidate"); + + subscribers[id] = res; + + req.on('close', function() { + delete subscribers[id]; + }); + +} + +function publish(message) { + + for (let id in subscribers) { + let res = subscribers[id]; + res.end(message); + } + + subscribers = Object.create(null); +} + +function accept(req, res) { + let urlParsed = url.parse(req.url, true); + + // new client wants messages + if (urlParsed.pathname == '/subscribe') { + onSubscribe(req, res); + return; + } + + // sending a message + if (urlParsed.pathname == '/publish' && req.method == 'POST') { + // accept POST + req.setEncoding('utf8'); + let message = ''; + req.on('data', function(chunk) { + message += chunk; + }).on('end', function() { + publish(message); // publish it to everyone + res.end("ok"); + }); + + return; + } + + // the rest is static + fileServer.serve(req, res); + +} + +function close() { + for (let id in subscribers) { + let res = subscribers[id]; + res.end(); + } +} + +// ----------------------------------- + +if (!module.parent) { + http.createServer(accept).listen(8080); + console.log('Server running on port 8080'); +} else { + exports.accept = accept; + + if (process.send) { + process.on('message', (msg) => { + if (msg === 'shutdown') { + close(); + } + }); + } + + process.on('SIGINT', close); +} diff --git a/5-network/08-websocket/article.md b/5-network/11-websocket/article.md similarity index 54% rename from 5-network/08-websocket/article.md rename to 5-network/11-websocket/article.md index 723922087..268b674f0 100644 --- a/5-network/08-websocket/article.md +++ b/5-network/11-websocket/article.md @@ -1,8 +1,6 @@ # WebSocket -The `WebSocket` protocol, described in the specification [RFC 6455](http://tools.ietf.org/html/rfc6455) provides a way to exchange data between browser and server via a persistent connection. - -Once a websocket connection is established, both client and server may send the data to each other. +The `WebSocket` protocol, described in the specification [RFC 6455](https://datatracker.ietf.org/doc/html/rfc6455), provides a way to exchange data between browser and server via a persistent connection. The data can be passed in both directions as "packets", without breaking the connection and the need of additional HTTP-requests. WebSocket is especially great for services that require continuous data exchange, e.g. online games, real-time trading systems and so on. @@ -17,11 +15,11 @@ let socket = new WebSocket("*!*ws*/!*://javascript.info"); There's also encrypted `wss://` protocol. It's like HTTPS for websockets. ```smart header="Always prefer `wss://`" -The `wss://` protocol not only encrypted, but also more reliable. +The `wss://` protocol is not only encrypted, but also more reliable. That's because `ws://` data is not encrypted, visible for any intermediary. Old proxy servers do not know about WebSocket, they may see "strange" headers and abort the connection. -On the other hand, `wss://` is WebSocket over TLS, (same as HTTPS is HTTP over TLS), the transport security layer encrypts the data at sender and decrypts at the receiver, so it passes encrypted through proxies. They can't see what's inside and let it through. +On the other hand, `wss://` is WebSocket over TLS, (same as HTTPS is HTTP over TLS), the transport security layer encrypts the data at the sender and decrypts it at the receiver. So data packets are passed encrypted through proxies. They can't see what's inside and let them through. ``` Once the socket is created, we should listen to events on it. There are totally 4 events: @@ -38,12 +36,13 @@ Here's an example: let socket = new WebSocket("wss://javascript.info/article/websocket/demo/hello"); socket.onopen = function(e) { - alert("[open] Connection established, send -> server"); + alert("[open] Connection established"); + alert("Sending to server"); socket.send("My name is John"); }; socket.onmessage = function(event) { - alert(`[message] Data received: ${event.data} <- server`); + alert(`[message] Data received from server: ${event.data}`); }; socket.onclose = function(event) { @@ -57,11 +56,11 @@ socket.onclose = function(event) { }; socket.onerror = function(error) { - alert(`[error] ${error.message}`); + alert(`[error]`); }; ``` -For demo purposes, there's a small server [server.js](demo/server.js) written in Node.js, for the example above, running. It responds with "hello", then waits 5 seconds and closes the connection. +For demo purposes, there's a small server [server.js](demo/server.js) written in Node.js, for the example above, running. It responds with "Hello from server, John", then waits 5 seconds and closes the connection. So you'll see events `open` -> `message` -> `close`. @@ -71,13 +70,13 @@ Now let's talk more in-depth. ## Opening a websocket -When `new WebSocket(url)` is created, it starts an HTTP handshake (HTTPS for `wss://`). +When `new WebSocket(url)` is created, it starts connecting immediately. -The browser asks the server: "Do you support Websocket?" And if the server says "yes", then the talk continues in WebSocket protocol, which is not HTTP at all. +During the connection, the browser (using headers) asks the server: "Do you support Websocket?" And if the server replies "yes", then the talk continues in WebSocket protocol, which is not HTTP at all.  -Here's an example of browser request for `new WebSocket("wss://javascript.info/chat")`. +Here's an example of browser headers for a request made by `new WebSocket("wss://javascript.info/chat")`. ``` GET /chat @@ -89,10 +88,10 @@ Sec-WebSocket-Key: Iv8io/9s+lYFgZWcXczP8Q== Sec-WebSocket-Version: 13 ``` -- `Origin` -- the origin of the client page. WebSocket is cross-origin by nature. There are no special headers or other limitations. Old servers are unable to handle WebSocket anyway, so there are no compabitility issues. But `Origin` header is important, as it allows the server to decide whether or not to talk WebSocket with this website. +- `Origin` -- the origin of the client page, e.g. `https://javascript.info`. WebSocket objects are cross-origin by nature. There are no special headers or other limitations. Old servers are unable to handle WebSocket anyway, so there are no compatibility issues. But the `Origin` header is important, as it allows the server to decide whether or not to talk WebSocket with this website. - `Connection: Upgrade` -- signals that the client would like to change the protocol. - `Upgrade: websocket` -- the requested protocol is "websocket". -- `Sec-WebSocket-Key` -- a random browser-generated key for security. +- `Sec-WebSocket-Key` -- a random browser-generated key, used to ensure that the server supports WebSocket protocol. It's random to prevent proxies from caching any following communication. - `Sec-WebSocket-Version` -- WebSocket protocol version, 13 is the current one. ```smart header="WebSocket handshake can't be emulated" @@ -108,9 +107,9 @@ Connection: Upgrade Sec-WebSocket-Accept: hsBlbuDTkk24srzEOTBUlZAlC2g= ``` -Here `Sec-WebSocket-Accept` is `Sec-WebSocket-Key`, recoded using a special algorithm. The browser uses it to make sure that the response corresponds to the request. +Here `Sec-WebSocket-Accept` is `Sec-WebSocket-Key`, recoded using a special algorithm. Upon seeing it, the browser understands that the server really does support the WebSocket protocol. -Afterwards, the data is transfered using WebSocket protocol, we'll see its structure ("frames") soon. And that's not HTTP at all. +Afterwards, the data is transferred using the WebSocket protocol, we'll see its structure ("frames") soon. And that's not HTTP at all. ### Extensions and subprotocols @@ -118,19 +117,17 @@ There may be additional headers `Sec-WebSocket-Extensions` and `Sec-WebSocket-Pr For instance: -- `Sec-WebSocket-Extensions: deflate-frame` means that the browser supports data compression. An extension is something related to transferring the data, not data itself. - -- `Sec-WebSocket-Protocol: soap, wamp` means that we're going to transfer not just any data, but the data in [SOAP](http://en.wikipedia.org/wiki/SOAP) or WAMP ("The WebSocket Application Messaging Protocol") protocols. WebSocket subprotocols are registered in the [IANA catalogue](http://www.iana.org/assignments/websocket/websocket.xml). +- `Sec-WebSocket-Extensions: deflate-frame` means that the browser supports data compression. An extension is something related to transferring the data, functionality that extends the WebSocket protocol. The header `Sec-WebSocket-Extensions` is sent automatically by the browser, with the list of all extensions it supports. -`Sec-WebSocket-Extensions` is sent by the browser automatically, with a list of possible extensions it supports. +- `Sec-WebSocket-Protocol: soap, wamp` means that we'd like to transfer not just any data, but the data in [SOAP](https://en.wikipedia.org/wiki/SOAP) or WAMP ("The WebSocket Application Messaging Protocol") protocols. WebSocket subprotocols are registered in the [IANA catalogue](https://www.iana.org/assignments/websocket/websocket.xml). So, this header describes the data formats that we're going to use. -`Sec-WebSocket-Protocol` is depends on us: we decide what kind of data we send. The second optional parameter of `new WebSocket` lists subprotocols: + This optional header is set using the second parameter of `new WebSocket`. That's the array of subprotocols, e.g. if we'd like to use SOAP or WAMP: -```js -let socket = new WebSocket("wss://javascript.info/chat", ["soap", "wamp"]); -``` + ```js + let socket = new WebSocket("wss://javascript.info/chat", ["soap", "wamp"]); + ``` -The server should respond with a list of protocls and extensions that it agrees to use. +The server should respond with a list of protocols and extensions that it agrees to use. For example, the request: @@ -161,31 +158,31 @@ Sec-WebSocket-Protocol: soap */!* ``` -Here the server responds that it supports the extension `deflate-frame`, and only SOAP of the requested subprotocols. +Here the server responds that it supports the extension "deflate-frame", and only SOAP of the requested subprotocols. -## WebSocket data +## Data transfer -WebSocket communication consists of "frames" that can be sent from either side: +WebSocket communication consists of "frames" -- data fragments, that can be sent from either side, and can be of several kinds: - "text frames" -- contain text data that parties send to each other. - "binary data frames" -- contain binary data that parties send to each other. - "ping/pong frames" are used to check the connection, sent from the server, the browser responds to these automatically. -- "connection close frame" and a few other service frames. +- there's also "connection close frame" and a few other service frames. -In the browser, we only care about text or binary frames. +In the browser, we directly work only with text or binary frames. -**WebSocket `.send()` can send either text or binary data, doesn't matter.** +**WebSocket `.send()` method can send either text or binary data.** -For sending, `socket.send(body)` allows strings or any binary format, including `Blob`, `ArrayBuffer`, etc. No settings required: just send it out. +A call `socket.send(body)` allows `body` in string or a binary format, including `Blob`, `ArrayBuffer`, etc. No settings are required: just send it out in any format. -**Textual data always comes as string. For receiving binary data, we can choose between `Blob` and `ArrayBuffer` formats.** +**When we receive the data, text always comes as string. And for binary data, we can choose between `Blob` and `ArrayBuffer` formats.** -The `socket.bufferType` is `"blob"` by default, so binary data comes in Blobs. +That's set by `socket.binaryType` property, it's `"blob"` by default, so binary data comes as `Blob` objects. [Blob](info:blob) is a high-level binary object, it directly integrates with `<a>`, `<img>` and other tags, so that's a sane default. But for binary processing, to access individual data bytes, we can change it to `"arraybuffer"`: ```js -socket.bufferType = "arraybuffer"; +socket.binaryType = "arraybuffer"; socket.onmessage = (event) => { // event.data is either a string (if text) or arraybuffer (if binary) }; @@ -193,16 +190,17 @@ socket.onmessage = (event) => { ## Rate limiting -Imagine, our app is generating a lot of data to send. But network connection is not that fast. The user may be on a mobile, in rural area. +Imagine, our app is generating a lot of data to send. But the user has a slow network connection, maybe on a mobile internet, outside of a city. -We can call `socket.send(data)` again and again. But the data will be buffered in memory and sent out only as fast as network speed allows. +We can call `socket.send(data)` again and again. But the data will be buffered (stored) in memory and sent out only as fast as network speed allows. -The `socket.bufferedAmount` property stores how many bytes are buffered at this moment, waiting to be sent over the network. +The `socket.bufferedAmount` property stores how many bytes remain buffered at this moment, waiting to be sent over the network. We can examine it to see whether the socket is actually available for transmission. ```js -// every 100ms examine the socket and send more data only if no data buffered +// every 100ms examine the socket and send more data +// only if all the existing data was sent out setInterval(() => { if (socket.bufferedAmount == 0) { socket.send(moreData()); @@ -215,18 +213,21 @@ setInterval(() => { Normally, when a party wants to close the connection (both browser and server have equal rights), they send a "connection close frame" with a numeric code and a textual reason. -The method is: +The method for that is: ```js socket.close([code], [reason]); ``` -Then the other party in `close` event handle can get the code and the reason, e.g.: +- `code` is a special WebSocket closing code (optional) +- `reason` is a string that describes the reason of closing (optional) + +Then the other party in the `close` event handler gets the code and the reason, e.g.: ```js -// one party: +// closing party: socket.close(1000, "Work complete"); -// another party +// the other party socket.onclose = event => { // event.code === 1000 // event.reason === "Work complete" @@ -234,12 +235,10 @@ socket.onclose = event => { }; ``` -The `code` is not just any number, but a special WebSocket closing code. - -Most common values: +Most common code values: -- `1000` -- the default, normal closure, -- `1006` -- can't set such code manually, indicates that the connection was broken (no close frame). +- `1000` -- the default, normal closure (used if no `code` supplied), +- `1006` -- no way to set such code manually, indicates that the connection was lost (no close frame). There are other codes like: @@ -248,9 +247,9 @@ There are other codes like: - `1011` -- unexpected error on server, - ...and so on. -Please refer to the [RFC6455, §7.4.1](https://tools.ietf.org/html/rfc6455#section-7.4.1) for the full list. +The full list can be found in [RFC6455, §7.4.1](https://tools.ietf.org/html/rfc6455#section-7.4.1). -WebSocket codes are somewhat like HTTP codes, but different. In particular, an codes less than `1000` are reserved, there'll be an error if we try to set such a code. +WebSocket codes are somewhat like HTTP codes, but different. In particular, codes lower than `1000` are reserved, there'll be an error if we try to set such a code. ```js // in case connection is broken @@ -274,9 +273,9 @@ To get connection state, additionally there's `socket.readyState` property with ## Chat example -Let's review a chat example using browser WebSocket API and Node.js WebSocket module <https://github.com/websockets/ws>. +Let's review a chat example using browser WebSocket API and Node.js WebSocket module <https://github.com/websockets/ws>. We'll pay the main attention to the client side, but the server is also simple. -HTML: there's a `<form>` to send messages and a `<div>` for incoming messages: +HTML: we need a `<form>` to send messages and a `<div>` for incoming messages: ```html <!-- message form --> @@ -289,7 +288,12 @@ HTML: there's a `<form>` to send messages and a `<div>` for incoming messages: <div id="messages"></div> ``` -JavaScript is also simple. We open a socket, then on form submission -- `socket.send(message)`, on incoming message -- append it to `div#messages`: +From JavaScript we want three things: +1. Open the connection. +2. On form submission -- `socket.send(message)` for the message. +3. On incoming message -- append it to `div#messages`. + +Here's the code: ```js let socket = new WebSocket("wss://javascript.info/article/websocket/chat/ws"); @@ -302,7 +306,7 @@ document.forms.publish.onsubmit = function() { return false; }; -// show message in div#messages +// message received - show the message in div#messages socket.onmessage = function(event) { let message = event.data; @@ -312,14 +316,13 @@ socket.onmessage = function(event) { } ``` -Server-side code is a little bit beyound our scope here. We're using browser WebSocket API, a server may have another library. +Server-side code is a little bit beyond our scope. Here we'll use Node.js, but you don't have to. Other platforms also have their means to work with WebSocket. -Still it can also be pretty simple. We'll use Node.js with <https://github.com/websockets/ws> module for websockets. +The server-side algorithm will be: -The algorithm will be: 1. Create `clients = new Set()` -- a set of sockets. -2. For each accepted websocket, `clients.add(socket)` and listen for its messages. -3. When a message received: iterate over clients and send it to everyone. +2. For each accepted websocket, add it to the set `clients.add(socket)` and set `message` event listener to get its messages. +3. When a message is received: iterate over clients and send it to everyone. 4. When a connection is closed: `clients.delete(socket)`. ```js @@ -329,7 +332,8 @@ const wss = new ws.Server({noServer: true}); const clients = new Set(); http.createServer((req, res) => { - // in real project we have additional code here to handle non-websocket requests + // here we only handle websocket connections + // in real project we'd have some other code here to handle non-websocket requests wss.handleUpgrade(req, req.socket, Buffer.alloc(0), onSocketConnect); }); @@ -355,12 +359,11 @@ Here's the working example: [iframe src="chat" height="100" zip] -You can also download it (upper-right button in the iframe) and run locally. Just don't forget to install [Node.js](https://nodejs.org/en/) and `npm install ws` before running. - +You can also download it (upper-right button in the iframe) and run it locally. Just don't forget to install [Node.js](https://nodejs.org/en/) and `npm install ws` before running. ## Summary -WebSocket is a modern way to have persisten browser-server connections. +WebSocket is a modern way to have persistent browser-server connections. - WebSockets don't have cross-origin limitations. - They are well-supported in browsers. @@ -378,8 +381,8 @@ Events: - `error`, - `close`. -WebSocket by itself does not include reconnection, authentication and many other high-level mechanisms. So there are client/server libraries add them. But it's also possible to implement these manually and integrate WebSockets with an existing site. +WebSocket by itself does not include reconnection, authentication and many other high-level mechanisms. So there are client/server libraries for that, and it's also possible to implement these capabilities manually. -For integration purposes, a WebSocket server is usually running in parallel with the main server, and they share a single database. Requests to WebSocket use `wss://ws.site.com`, a subdomain that leads to WebSocket server, while `https://site.com` goes to the main HTTP-server. +Sometimes, to integrate WebSocket into existing projects, people run a WebSocket server in parallel with the main HTTP-server, and they share a single database. Requests to WebSocket use `wss://ws.site.com`, a subdomain that leads to the WebSocket server, while `https://site.com` goes to the main HTTP-server. -Surely, other ways of integration are also possible. Many servers (such as Node.js) can support both HTTP and WebSocket protocols. +Surely, other ways of integration are also possible. diff --git a/5-network/08-websocket/chat.view/index.html b/5-network/11-websocket/chat.view/index.html similarity index 100% rename from 5-network/08-websocket/chat.view/index.html rename to 5-network/11-websocket/chat.view/index.html diff --git a/5-network/08-websocket/chat.view/server.js b/5-network/11-websocket/chat.view/server.js similarity index 100% rename from 5-network/08-websocket/chat.view/server.js rename to 5-network/11-websocket/chat.view/server.js diff --git a/5-network/08-websocket/demo.view/server.js b/5-network/11-websocket/demo.view/server.js similarity index 74% rename from 5-network/08-websocket/demo.view/server.js rename to 5-network/11-websocket/demo.view/server.js index 7b17734d7..bff739938 100644 --- a/5-network/08-websocket/demo.view/server.js +++ b/5-network/11-websocket/demo.view/server.js @@ -9,8 +9,9 @@ function accept(req, res) { res.end(); return; } + // can be Connection: keep-alive, Upgrade - if (req.headers.connection.match(/\bupgrade\b/i)) { + if (!req.headers.connection.match(/\bupgrade\b/i)) { res.end(); return; } @@ -20,8 +21,9 @@ function accept(req, res) { function onConnect(ws) { ws.on('message', function (message) { - let name = message.match(/\w+$/) || "Guest"; - ws.send(`Hello, ${name}!`); + message = message.toString(); + let name = message.match(/([\p{Alpha}\p{M}\p{Nd}\p{Pc}\p{Join_C}]+)$/gu) || "Guest"; + ws.send(`Hello from server, ${name}!`); setTimeout(() => ws.close(1000, "Bye!"), 5000); }); diff --git a/5-network/08-websocket/websocket-handshake.svg b/5-network/11-websocket/websocket-handshake.svg similarity index 57% rename from 5-network/08-websocket/websocket-handshake.svg rename to 5-network/11-websocket/websocket-handshake.svg index b05713747..2c406b5a3 100644 --- a/5-network/08-websocket/websocket-handshake.svg +++ b/5-network/11-websocket/websocket-handshake.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:5-network/08-websocket/websocket-handshake.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="429px" height="404px" viewBox="0 0 429 404" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -31,4 +32,7 @@ </text> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="429" height="348" viewBox="0 0 429 348"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="network" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="websocket-handshake.svg"><path id="Rectangle-227" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M2 16h128v64H2z"/><text id="Browser" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="40" y="49">Browser</tspan></text><path id="Rectangle-228" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2" d="M298 16h128v64H298z"/><text id="Server" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="340" y="49">Server</tspan></text><path id="Line" stroke="#7E7C7B" stroke-linecap="square" d="M67 81v250.5M363 81v251.5"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M349 133l14 7-14 7v-6H68v-2h281v-6z"/><path id="Line-Copy" fill="#C06334" fill-rule="nonzero" d="M83 204v6h281v2H83v6l-14-7 14-7z"/><text id="HTTP-request" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="172.015" y="130">HTTP-request</tspan></text><text id=""Hey,-server,-let's" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="102.605" y="161">"Hey, server, let's talk WebSocket?"</tspan></text><text id="HTTP-response-"Okay!" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="166.419" y="204">HTTP-response</tspan> <tspan x="191.972" y="226">"Okay!"</tspan></text><path id="Line-Copy-2" fill="#C06334" fill-rule="nonzero" d="M81 272v6h2v2h-2v6l-14-7 14-7zm268 0l14 7-14 7v-14zm-254 6v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm-126 0v2h-4v-2h4zm138 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm6 0v2h-4v-2h4zm-126 0v2h-4v-2h4z" transform="matrix(-1 0 0 1 430 0)"/><text id="WebSocket-protocol" fill="#AF6E24" font-family="OpenSans-Regular, Open Sans" font-size="14" font-weight="normal"><tspan x="151.604" y="272">WebSocket protocol</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:5-network/11-websocket/websocket-handshake.svg diff --git a/5-network/12-server-sent-events/article.md b/5-network/12-server-sent-events/article.md new file mode 100644 index 000000000..c59d671a4 --- /dev/null +++ b/5-network/12-server-sent-events/article.md @@ -0,0 +1,271 @@ +# Server Sent Events + +The [Server-Sent Events](https://html.spec.whatwg.org/multipage/comms.html#the-eventsource-interface) specification describes a built-in class `EventSource`, that keeps connection with the server and allows to receive events from it. + +Similar to `WebSocket`, the connection is persistent. + +But there are several important differences: + +| `WebSocket` | `EventSource` | +|-------------|---------------| +| Bi-directional: both client and server can exchange messages | One-directional: only server sends data | +| Binary and text data | Only text | +| WebSocket protocol | Regular HTTP | + +`EventSource` is a less-powerful way of communicating with the server than `WebSocket`. + +Why should one ever use it? + +The main reason: it's simpler. In many applications, the power of `WebSocket` is a little bit too much. + +We need to receive a stream of data from server: maybe chat messages or market prices, or whatever. That's what `EventSource` is good at. Also it supports auto-reconnect, something we need to implement manually with `WebSocket`. Besides, it's a plain old HTTP, not a new protocol. + +## Getting messages + +To start receiving messages, we just need to create `new EventSource(url)`. + +The browser will connect to `url` and keep the connection open, waiting for events. + +The server should respond with status 200 and the header `Content-Type: text/event-stream`, then keep the connection and write messages into it in the special format, like this: + +``` +data: Message 1 + +data: Message 2 + +data: Message 3 +data: of two lines +``` + +- A message text goes after `data:`, the space after the colon is optional. +- Messages are delimited with double line breaks `\n\n`. +- To send a line break `\n`, we can immediately send one more `data:` (3rd message above). + +In practice, complex messages are usually sent JSON-encoded. Line-breaks are encoded as `\n` within them, so multiline `data:` messages are not necessary. + +For instance: + +```js +data: {"user":"John","message":"First line*!*\n*/!* Second line"} +``` + +...So we can assume that one `data:` holds exactly one message. + +For each such message, the `message` event is generated: + +```js +let eventSource = new EventSource("/events/subscribe"); + +eventSource.onmessage = function(event) { + console.log("New message", event.data); + // will log 3 times for the data stream above +}; + +// or eventSource.addEventListener('message', ...) +``` + +### Cross-origin requests + +`EventSource` supports cross-origin requests, like `fetch` and any other networking methods. We can use any URL: + +```js +let source = new EventSource("https://another-site.com/events"); +``` + +The remote server will get the `Origin` header and must respond with `Access-Control-Allow-Origin` to proceed. + +To pass credentials, we should set the additional option `withCredentials`, like this: + +```js +let source = new EventSource("https://another-site.com/events", { + withCredentials: true +}); +``` + +Please see the chapter <info:fetch-crossorigin> for more details about cross-origin headers. + + +## Reconnection + +Upon creation, `new EventSource` connects to the server, and if the connection is broken -- reconnects. + +That's very convenient, as we don't have to care about it. + +There's a small delay between reconnections, a few seconds by default. + +The server can set the recommended delay using `retry:` in response (in milliseconds): + +```js +retry: 15000 +data: Hello, I set the reconnection delay to 15 seconds +``` + +The `retry:` may come both together with some data, or as a standalone message. + +The browser should wait that many milliseconds before reconnecting. Or longer, e.g. if the browser knows (from OS) that there's no network connection at the moment, it may wait until the connection appears, and then retry. + +- If the server wants the browser to stop reconnecting, it should respond with HTTP status 204. +- If the browser wants to close the connection, it should call `eventSource.close()`: + +```js +let eventSource = new EventSource(...); + +eventSource.close(); +``` + +Also, there will be no reconnection if the response has an incorrect `Content-Type` or its HTTP status differs from 301, 307, 200 and 204. In such cases the `"error"` event will be emitted, and the browser won't reconnect. + +```smart +When a connection is finally closed, there's no way to "reopen" it. If we'd like to connect again, just create a new `EventSource`. +``` + +## Message id + +When a connection breaks due to network problems, either side can't be sure which messages were received, and which weren't. + +To correctly resume the connection, each message should have an `id` field, like this: + +``` +data: Message 1 +id: 1 + +data: Message 2 +id: 2 + +data: Message 3 +data: of two lines +id: 3 +``` + +When a message with `id:` is received, the browser: + +- Sets the property `eventSource.lastEventId` to its value. +- Upon reconnection sends the header `Last-Event-ID` with that `id`, so that the server may re-send following messages. + +```smart header="Put `id:` after `data:`" +Please note: the `id` is appended below message `data` by the server, to ensure that `lastEventId` is updated after the message is received. +``` + +## Connection status: readyState + +The `EventSource` object has `readyState` property, that has one of three values: + +```js no-beautify +EventSource.CONNECTING = 0; // connecting or reconnecting +EventSource.OPEN = 1; // connected +EventSource.CLOSED = 2; // connection closed +``` + +When an object is created, or the connection is down, it's always `EventSource.CONNECTING` (equals `0`). + +We can query this property to know the state of `EventSource`. + +## Event types + +By default `EventSource` object generates three events: + +- `message` -- a message received, available as `event.data`. +- `open` -- the connection is open. +- `error` -- the connection could not be established, e.g. the server returned HTTP 500 status. + +The server may specify another type of event with `event: ...` at the event start. + +For example: + +``` +event: join +data: Bob + +data: Hello + +event: leave +data: Bob +``` + +To handle custom events, we must use `addEventListener`, not `onmessage`: + +```js +eventSource.addEventListener('join', event => { + alert(`Joined ${event.data}`); +}); + +eventSource.addEventListener('message', event => { + alert(`Said: ${event.data}`); +}); + +eventSource.addEventListener('leave', event => { + alert(`Left ${event.data}`); +}); +``` + +## Full example + +Here's the server that sends messages with `1`, `2`, `3`, then `bye` and breaks the connection. + +Then the browser automatically reconnects. + +[codetabs src="eventsource"] + +## Summary + +`EventSource` object automatically establishes a persistent connection and allows the server to send messages over it. + +It offers: +- Automatic reconnect, with tunable `retry` timeout. +- Message ids to resume events, the last received identifier is sent in `Last-Event-ID` header upon reconnection. +- The current state is in the `readyState` property. + +That makes `EventSource` a viable alternative to `WebSocket`, as the latter is more low-level and lacks such built-in features (though they can be implemented). + +In many real-life applications, the power of `EventSource` is just enough. + +Supported in all modern browsers (not IE). + +The syntax is: + +```js +let source = new EventSource(url, [credentials]); +``` + +The second argument has only one possible option: `{ withCredentials: true }`, it allows sending cross-origin credentials. + +Overall cross-origin security is same as for `fetch` and other network methods. + +### Properties of an `EventSource` object + +`readyState` +: The current connection state: either `EventSource.CONNECTING (=0)`, `EventSource.OPEN (=1)` or `EventSource.CLOSED (=2)`. + +`lastEventId` +: The last received `id`. Upon reconnection the browser sends it in the header `Last-Event-ID`. + +### Methods + +`close()` +: Closes the connection. + +### Events + +`message` +: Message received, the data is in `event.data`. + +`open` +: The connection is established. + +`error` +: In case of an error, including both lost connection (will auto-reconnect) and fatal errors. We can check `readyState` to see if the reconnection is being attempted. + +The server may set a custom event name in `event:`. Such events should be handled using `addEventListener`, not `on<event>`. + +### Server response format + +The server sends messages, delimited by `\n\n`. + +A message may have following fields: + +- `data:` -- message body, a sequence of multiple `data` is interpreted as a single message, with `\n` between the parts. +- `id:` -- renews `lastEventId`, sent in `Last-Event-ID` on reconnect. +- `retry:` -- recommends a retry delay for reconnections in ms. There's no way to set it from JavaScript. +- `event:` -- event name, must precede `data:`. + +A message may include one or more fields in any order, but `id:` usually goes the last. diff --git a/5-network/12-server-sent-events/eventsource.view/index.html b/5-network/12-server-sent-events/eventsource.view/index.html new file mode 100644 index 000000000..795b07ebb --- /dev/null +++ b/5-network/12-server-sent-events/eventsource.view/index.html @@ -0,0 +1,50 @@ +<!DOCTYPE html> +<script> +let eventSource; + +function start() { // when "Start" button pressed + if (!window.EventSource) { + // IE or an old browser + alert("The browser doesn't support EventSource."); + return; + } + + eventSource = new EventSource('digits'); + + eventSource.onopen = function(e) { + log("Event: open"); + }; + + eventSource.onerror = function(e) { + log("Event: error"); + if (this.readyState == EventSource.CONNECTING) { + log(`Reconnecting (readyState=${this.readyState})...`); + } else { + log("Error has occured."); + } + }; + + eventSource.addEventListener('bye', function(e) { + log("Event: bye, data: " + e.data); + }); + + eventSource.onmessage = function(e) { + log("Event: message, data: " + e.data); + }; +} + +function stop() { // when "Stop" button pressed + eventSource.close(); + log("eventSource.close()"); +} + +function log(msg) { + logElem.innerHTML += msg + "<br>"; + document.documentElement.scrollTop = 99999999; +} +</script> + +<button onclick="start()">Start</button> Press the "Start" to begin. +<div id="logElem" style="margin: 6px 0"></div> + +<button onclick="stop()">Stop</button> "Stop" to finish. diff --git a/5-network/12-server-sent-events/eventsource.view/server.js b/5-network/12-server-sent-events/eventsource.view/server.js new file mode 100644 index 000000000..ad279aaab --- /dev/null +++ b/5-network/12-server-sent-events/eventsource.view/server.js @@ -0,0 +1,48 @@ +let http = require('http'); +let url = require('url'); +let querystring = require('querystring'); +let static = require('node-static'); +let fileServer = new static.Server('.'); + +function onDigits(req, res) { + res.writeHead(200, { + 'Content-Type': 'text/event-stream; charset=utf-8', + 'Cache-Control': 'no-cache' + }); + + let i = 0; + + let timer = setInterval(write, 1000); + write(); + + function write() { + i++; + + if (i == 4) { + res.write('event: bye\ndata: bye-bye\n\n'); + clearInterval(timer); + res.end(); + return; + } + + res.write('data: ' + i + '\n\n'); + + } +} + +function accept(req, res) { + + if (req.url == '/digits') { + onDigits(req, res); + return; + } + + fileServer.serve(req, res); +} + + +if (!module.parent) { + http.createServer(accept).listen(8080); +} else { + exports.accept = accept; +} diff --git a/6-data-storage/01-cookie/article.md b/6-data-storage/01-cookie/article.md index ce6e17dbb..1b9e93414 100644 --- a/6-data-storage/01-cookie/article.md +++ b/6-data-storage/01-cookie/article.md @@ -1,27 +1,27 @@ # Cookies, document.cookie -Cookies are small strings of data that are stored directly in the browser. They are a part of HTTP protocol, defined by [RFC 6265](https://tools.ietf.org/html/rfc6265) specification. +Cookies are small strings of data that are stored directly in the browser. They are a part of the HTTP protocol, defined by the [RFC 6265](https://tools.ietf.org/html/rfc6265) specification. -Most of the time, cookies are set by a web server. Then they are automatically added to every request to the same domain. +Cookies are usually set by a web server using the response `Set-Cookie` HTTP header. Then, the browser automatically adds them to (almost) every request to the same domain using the `Cookie` HTTP header. One of the most widespread use cases is authentication: -1. Upon sign in, the server uses `Set-Cookie` HTTP-header in the response to set a cookie with "session identifier". -2. Next time when the request is set to the same domain, the browser sends the over the net using `Cookie` HTTP-header. +1. Upon sign-in, the server uses the `Set-Cookie` HTTP header in the response to set a cookie with a unique "session identifier". +2. Next time the request is sent to the same domain, the browser sends the cookie over the net using the `Cookie` HTTP header. 3. So the server knows who made the request. We can also access cookies from the browser, using `document.cookie` property. -There are many tricky things about cookies and their options. In this chapter we'll cover them in detail. +There are many tricky things about cookies and their attributes. In this chapter, we'll cover them in detail. ## Reading from document.cookie ```online -Do you have any cookies on this site? Let's see: +Does your browser store any cookies from this site? Let's see: ``` ```offline -Assuming you're on a website, it's possible to see the cookies, like this: +Assuming you're on a website, it's possible to see the cookies from it, like this: ``` ```js run @@ -31,17 +31,17 @@ alert( document.cookie ); // cookie1=value1; cookie2=value2;... ``` -The value of `document.cookie` consists of `name=value` pairs, delimited by `; `. Each one is a separate cookie. +The value of `document.cookie` consists of `name=value` pairs, delimited by `; `. Each one is a separate cookie. -To find a particular cookie, we can split `document.cookie` by `; `, and then find the right name. We can use either a regular expression or array functions to do that. +To find a particular cookie, we can split `document.cookie` by `; `, and then find the right name. We can use either a regular expression or array functions to do that. -We leave it as an exercise for the reader. Also, at the end of the chapter you'll find helper functions to manipulate cookies. +We leave it as an exercise for the reader. Also, at the end of the chapter, you'll find helper functions to manipulate cookies. ## Writing to document.cookie -We can write to `document.cookie`. But it's not a data property, it's an accessor. +We can write to `document.cookie`. But it's not a data property, it's an [accessor (getter/setter)](info:property-accessors). An assignment to it is treated specially. -**A write operation to `document.cookie` passes through the browser that updates cookies mentioned in it, but doesn't touch other cookies.** +**A write operation to `document.cookie` updates only the cookie mentioned in it and doesn't touch other cookies.** For instance, this call sets a cookie with the name `user` and value `John`: @@ -50,12 +50,12 @@ document.cookie = "user=John"; // update only cookie named 'user' alert(document.cookie); // show all cookies ``` -If you run it, then probably you'll see multiple cookies. That's because `document.cookie=` operation does not overwrite all cookies. It only sets the mentioned cookie `user`. +If you run it, you will likely see multiple cookies. That's because the `document.cookie=` operation does not overwrite all cookies. It only sets the mentioned cookie `user`. -Technically, name and value can have any characters, but to keep the formatting valid they should be escaped using a built-in `encodeURIComponent` function: +Technically, name and value can have any characters. To keep the valid formatting, they should be escaped using a built-in `encodeURIComponent` function: ```js run -// special values, need encoding +// special characters (spaces) need encoding let name = "my name"; let value = "John Smith" @@ -67,76 +67,86 @@ alert(document.cookie); // ...; my%20name=John%20Smith ```warn header="Limitations" -There are few limitations: -- The `name=value` pair, after `encodeURIComponent`, should not exceed 4kb. So we can't store anything huge in a cookie. -- The total number of cookies per domain is limited to around 20+, the exact limit depends on a browser. +There are a few limitations: +- You can only set/update a single cookie at a time using `document.cookie`. +- The `name=value` pair, after `encodeURIComponent`, should not exceed 4KB. So we can't store anything huge in a cookie. +- The total number of cookies per domain is limited to around 20+, the exact limit depends on the browser. ``` -Cookies have several options, many of them are important and should be set. +Cookies have several attributes, many of which are important and should be set. -The options are listed after `key=value`, delimited by `;`, like this: +The attributes are listed after `key=value`, delimited by `;`, like this: ```js run document.cookie = "user=John; path=/; expires=Tue, 19 Jan 2038 03:14:07 GMT" ``` -## path - -- **`path=/mypath`** - -The url path prefix, where the cookie is accessible. Must be absolute. By default, it's the current path. - -If a cookie is set with `path=/admin`, it's visible at pages `/admin` and `/admin/something`, but not at `/home` or `/adminpage`. - -Usually, we set `path=/` to make the cookie accessible from all website pages. - ## domain - **`domain=site.com`** -A domain where the cookie is accessible. In practice though, there are limitations. We can't set any domain. +A domain defines where the cookie is accessible. In practice though, there are limitations. We can't set any domain. -By default, a cookie is accessible only at the domain that set it. So, if the cookie was set by `site.com`, we won't get it `other.com`. +**There's no way to let a cookie be accessible from another 2nd-level domain, so `other.com` will never receive a cookie set at `site.com`.** + +It's a safety restriction, to allow us to store sensitive data in cookies that should be available only on one site. -...But what's more tricky, we also won't get the cookie at a subdomain `forum.site.com`! +By default, a cookie is accessible only at the domain that set it. + +Please note, by default, a cookie is not shared with a subdomain, such as `forum.site.com`. ```js -// at site.com +// if we set a cookie at site.com website... document.cookie = "user=John" -// at forum.site.com +// ...we won't see it at forum.site.com alert(document.cookie); // no user ``` -**There's no way to let a cookie be accessible from another 2nd-level domain, so `other.com` will never receive a cookie set at `site.com`.** +...But this can be changed. If we'd like to allow subdomains like `forum.site.com` to get a cookie set at `site.com`, that's possible. -It's a safety restriction, to allow us to store sensitive data in cookies. +For that to happen, when setting a cookie at `site.com`, we should explicitly set the `domain` attribute to the root domain: `domain=site.com`. Then all subdomains will see such a cookie. -...But if we'd like to grant access to subdomains like `forum.site.com`, that's possible. We should explicitly set `domain` option to the root domain: `domain=site.com`: +For example: ```js -// at site.com, make the cookie accessible on any subdomain: -document.cookie = "user=John; domain=site.com" +// at site.com +// make the cookie accessible on any subdomain *.site.com: +document.cookie = "user=John; *!*domain=site.com*/!*" + +// later // at forum.site.com -alert(document.cookie); // with user +alert(document.cookie); // has cookie user=John +``` + +```warn header="Legacy syntax" +Historically, `domain=.site.com` (with a dot before `site.com`) used to work the same way, allowing access to the cookie from subdomains. Leading dots in domain names are now ignored, but some browsers may decline to set the cookie containing such dots. ``` -For historical reasons, `domain=.site.com` (a dot at the start) also works this way, it might better to add the dot to support very old browsers. +To summarize, the `domain` attribute allows to make a cookie accessible at subdomains. + +## path + +- **`path=/mypath`** + +The URL path prefix must be absolute. It makes the cookie accessible for pages under that path. By default, it's the current path. + +If a cookie is set with `path=/admin`, it's visible on pages `/admin` and `/admin/something`, but not at `/home`, `/home/admin` or `/`. -So, `domain` option allows to make a cookie accessible at subdomains. +Usually, we should set `path` to the root: `path=/` to make the cookie accessible from all website pages. If this attribute is not set the default is calculated using [this method](https://developer.mozilla.org/en-US/docs/Web/HTTP/Cookies#path_default_value). ## expires, max-age -By default, if a cookie doesn't have one of these options, it disappears when the browser is closed. Such cookies are called "session cookies" +By default, if a cookie doesn't have one of these attributes, it disappears when the browser/tab is closed. Such cookies are called "session cookies" -To let cookies survive browser close, we can set either `expires` or `max-age` option. +To let cookies survive a browser close, we can set either the `expires` or `max-age` attribute. `max-Age` has precedence if both are set. - **`expires=Tue, 19 Jan 2038 03:14:07 GMT`** -Cookie expiration date, when the browser will delete it automatically. +The cookie expiration date defines the time when the browser will automatically delete it (according to the browser's time zone). -The date must be exactly in this format, in GMT timezone. We can use `date.toUTCString` to get it. For instance, we can set the cookie to expire in 1 day: +The date must be exactly in this format, in the GMT timezone. We can use `date.toUTCString` to get it. For instance, we can set the cookie to expire in 1 day: ```js // +1 day from now @@ -149,17 +159,17 @@ If we set `expires` to a date in the past, the cookie is deleted. - **`max-age=3600`** -An alternative to `expires`, specifies the cookie expiration in seconds from the current moment. +It's an alternative to `expires` and specifies the cookie's expiration in seconds from the current moment. -If zero or negative, then the cookie is deleted: +If set to zero or a negative value, the cookie is deleted: ```js -// cookie will die +1 hour from now +// cookie will die in +1 hour from now document.cookie = "user=John; max-age=3600"; // delete cookie (let it expire right now) document.cookie = "user=John; max-age=0"; -``` +``` ## secure @@ -171,98 +181,101 @@ The cookie should be transferred only over HTTPS. That is, cookies are domain-based, they do not distinguish between the protocols. -With this option, if a cookie is set by `https://site.com`, then it doesn't appear when the same site is accessed by HTTP, as `http://site.com`. So if a cookie has sensitive content that should never be sent over unencrypted HTTP, then the flag is the right thing. +With this attribute, if a cookie is set by `https://site.com`, then it doesn't appear when the same site is accessed by HTTP, as `http://site.com`. So if a cookie has sensitive content that should never be sent over unencrypted HTTP, the `secure` flag is the right thing. ```js // assuming we're on https:// now -// set the cookie secure (only accessible if over HTTPS) +// set the cookie to be secure (only accessible over HTTPS) document.cookie = "user=John; secure"; -``` +``` ## samesite -That's another security option, to protect from so-called XSRF (cross-site request forgery) attacks. +This is another security attribute `samesite`. It's designed to protect from so-called XSRF (cross-site request forgery) attacks. -To understand when it's useful, let's introduce the following attack scenario. +To understand how it works and when it's useful, let's take a look at XSRF attacks. ### XSRF attack -Imagine, you are logged into the site `bank.com`. That is: you have an authentication cookie from that site. Your browser sends it to `bank.com` with every request, so that it recognizes you and performs all sensitive financial operations. +Imagine, you are logged into the site `bank.com`. That is: you have an authentication cookie from that site. Your browser sends it to `bank.com` with every request so that it recognizes you and performs all sensitive financial operations. -Now, while browsing the web in another window, you occasionally come to another site `evil.com`, that automatically submits a form `<form action="https://bank.com/pay">` to `bank.com` with input fields that initiate a transaction to the hacker's account. +Now, while browsing the web in another window, you accidentally come to another site `evil.com`. That site has JavaScript code that submits a form `<form action="https://bank.com/pay">` to `bank.com` with fields that initiate a transaction to the hacker's account. -The form is submitted from `evil.com` directly to the bank site, and your cookie is also sent, just because it's sent every time you visit `bank.com`. So the bank recognizes you and actually performs the payment. +The browser sends cookies every time you visit the site `bank.com`, even if the form was submitted from `evil.com`. So the bank recognizes you and performs the payment.  -That's called a cross-site request forgery (or XSRF) attack. +This is a so-called "Cross-Site Request Forgery" (in short, XSRF) attack. -Real banks are protected from it of course. All forms generated by `bank.com` have a special field, so called "xsrf protection token", that an evil page can't neither generate, nor somehow extract from a remote page (it can submit a form there, but can't get the data back). +Real banks are protected from it of course. All forms generated by `bank.com` have a special field, a so-called "XSRF protection token", that an evil page can't generate or extract from a remote page. It can submit a form there, but can't get the data back. The site `bank.com` checks for such a token in every form it receives. -But that takes time to implement: we need to ensure that every form has the token field, and we must also check all requests. +Such a protection takes time to implement though. We need to ensure that every form has the required token field, and we must also check all requests. -### Enter cookie samesite option +### Use cookie samesite attribute -The cookie `samesite` option provides another way to protect from such attacks, that (in theory) should not require "xsrf protection tokens". +The cookie `samesite` attribute provides another way to protect from such attacks, that (in theory) should not require "xsrf protection tokens". It has two possible values: -- **`samesite=strict` (same as `samesite` without value)** +- **`samesite=strict`** -A cookie with `samesite=strict` is never sent if the user comes from outside the site. +A cookie with `samesite=strict` is never sent if the user comes from outside the same site. -In other words, whether a user follows a link from their mail or submits a form from `evil.com`, or does any operation that originates from another domain, the cookie is not sent. +In other words, whether a user follows a link from their email, submits a form from `evil.com`, or does any operation that originates from another domain, the cookie is not sent. -If authentication cookies have `samesite` option, then XSRF attack has no chances to succeed, because a submission from `evil.com` comes without cookies. So `bank.com` will not recognize the user and will not proceed with the payment. +If authentication cookies have the `samesite=strict` attribute, then an XSRF attack has no chance of succeeding, because a submission from `evil.com` comes without cookies. So `bank.com` will not recognize the user and will not proceed with the payment. -The protection is quite reliable. Only operations that come from `bank.com` will send the `samesite` cookie. +The protection is quite reliable. Only operations that come from `bank.com` will send the `samesite=strict` cookie, e.g. a form submission from another page at `bank.com`. Although, there's a small inconvenience. -When a user follows a legitimate link to `bank.com`, like from their own notes, they'll be surprised that `bank.com` does not recognize them. Indeed, `samesite=strict` cookies are not sent in that case. +When a user follows a legitimate link to `bank.com`, like from their notes, they'll be surprised that `bank.com` does not recognize them. Indeed, `samesite=strict` cookies are not sent in that case. -We could work around that by using two cookies: one for "general recognition", only for the purposes of saying: "Hello, John", and the other one for data-changing operations with `samesite=strict`. Then a person coming from outside of the site will see a welcome, but payments must be initiated from the bank website. +We could work around that by using two cookies: one for "general recognition", only to say: "Hello, John", and the other one for data-changing operations with `samesite=strict`. Then, a person coming from outside of the site will see a welcome, but payments must be initiated from the bank's website, for the second cookie to be sent. -- **`samesite=lax`** +- **`samesite=lax` (same as `samesite` without value)** -A more relaxed approach that also protects from XSRF and doesn't break user experience. +A more relaxed approach that also protects from XSRF and doesn't break the user experience. Lax mode, just like `strict`, forbids the browser to send cookies when coming from outside the site, but adds an exception. A `samesite=lax` cookie is sent if both of these conditions are true: 1. The HTTP method is "safe" (e.g. GET, but not POST). - The full list safe of HTTP methods is in the [RFC7231 specification](https://tools.ietf.org/html/rfc7231). Basically, these are the methods that should be used for reading, but not writing the data. They must not perform any data-changing operations. Following a link is always GET, the safe method. + The full list of safe HTTP methods is in the [RFC7231 specification](https://tools.ietf.org/html/rfc7231#section-4.2.1). These are the methods that should be used for reading, but not writing the data. They must not perform any data-changing operations. Following a link is always GET, the safe method. -2. The operation performs top-level navigation (changes URL in the browser address bar). +2. The operation performs a top-level navigation (changes URL in the browser address bar). - That's usually true, but if the navigation is performed in an `<iframe>`, then it's not top-level. Also, AJAX requests do not perform any navigation, hence they don't fit. + This is usually true, but if the navigation is performed in an `<iframe>`, then it is not top-level. Additionally, JavaScript methods for network requests do not perform any navigation. -So, what `samesite=lax` does is basically allows a most common "go to URL" operation to have cookies. E.g. opening a website link from notes satisfies these conditions. +So, what `samesite=lax` does, is to allow the most common "go to URL" operation to have cookies. E.g. opening a website link from notes that satisfy these conditions. -But anything more complicated, like AJAX request from another site or a form submittion loses cookies. +But anything more complicated, like a network request from another site or a form submission, loses cookies. If that's fine for you, then adding `samesite=lax` will probably not break the user experience and add protection. -Overall, `samesite` is a great option, but it has an important drawback: -- `samesite` is ignored (not supported) by old browsers, year 2017 or so. +Overall, `samesite` is a great attribute. + +There's a drawback: + +- `samesite` is ignored (not supported) by very old browsers, the year 2017 or so. **So if we solely rely on `samesite` to provide protection, then old browsers will be vulnerable.** -But we surely can use `samesite` together with other protection measures, like xsrf tokens, to add an additional layer of defence and then, in the future, when old browsers die out, we'll probably be able to drop xsrf tokens. +But we can use `samesite` together with other protection measures, like xsrf tokens, to add a layer of defence and then, in the future, when old browsers die out, we'll probably be able to drop xsrf tokens. ## httpOnly -This option has nothing to do with JavaScript, but we have to mention it for completeness. +This attribute has nothing to do with JavaScript, but we have to mention it for completeness. -The web-server uses `Set-Cookie` header to set a cookie. And it may set the `httpOnly` option. +The web server uses the `Set-Cookie` header to set a cookie. Also, it may set the `httpOnly` attribute. -This option forbids any JavaScript access to the cookie. We can't see such cookie or manipulate it using `document.cookie`. +This attribute forbids any JavaScript access to the cookie. We can't see such a cookie or manipulate it using `document.cookie`. -That's used as a precaution measure, to protect from certain attacks when a hacker injects his own JavaScript code into a page and waits for a user to visit that page. That shouldn't be possible at all, a hacker should not be able to inject their code into our site, but there may be bugs that let hackers do it. +This is used as a precautionary measure, to protect from certain attacks when a hacker injects his own JavaScript code into a page and waits for a user to visit that page. That shouldn't be possible at all, hackers should not be able to inject their code into our site, but there may be bugs that let them do it. -Normally, if such thing happens, and a user visits a web-page with hacker's code, then that code executes and gains access to `document.cookie` with user cookies containing authentication information. That's bad. +Normally, if such a thing happens, and a user visits a web-page with a hacker's JavaScript code, then that code executes and gains access to `document.cookie` with user cookies containing authentication information. That's bad. But if a cookie is `httpOnly`, then `document.cookie` doesn't see it, so it is protected. @@ -272,10 +285,9 @@ Here's a small set of functions to work with cookies, more convenient than a man There exist many cookie libraries for that, so these are for demo purposes. Fully working though. - ### getCookie(name) -The shortest way to access cookie is to use a [regular expression](info:regular-expressions). +The shortest way to access a cookie is to use a [regular expression](info:regular-expressions). The function `getCookie(name)` returns the cookie with the given `name`: @@ -294,30 +306,30 @@ Here `new RegExp` is generated dynamically, to match `; name=<value>`. Please note that a cookie value is encoded, so `getCookie` uses a built-in `decodeURIComponent` function to decode it. -### setCookie(name, value, options) +### setCookie(name, value, attributes) -Sets the cookie `name` to the given `value` with `path=/` by default (can be modified to add other defaults): +Sets the cookie's `name` to the given `value` with `path=/` by default (can be modified to add other defaults): ```js run -function setCookie(name, value, options = {}) { +function setCookie(name, value, attributes = {}) { - options = { + attributes = { path: '/', // add other defaults here if necessary - ...options + ...attributes }; - if (options.expires.toUTCString) { - options.expires = options.expires.toUTCString(); + if (attributes.expires instanceof Date) { + attributes.expires = attributes.expires.toUTCString(); } let updatedCookie = encodeURIComponent(name) + "=" + encodeURIComponent(value); - for (let optionKey in options) { - updatedCookie += "; " + optionKey; - let optionValue = options[optionKey]; - if (optionValue !== true) { - updatedCookie += "=" + optionValue; + for (let attributeKey in attributes) { + updatedCookie += "; " + attributeKey; + let attributeValue = attributes[attributeKey]; + if (attributeValue !== true) { + updatedCookie += "=" + attributeValue; } } @@ -341,7 +353,7 @@ function deleteCookie(name) { ``` ```warn header="Updating or deleting must use same path and domain" -Please note: when we update or delete a cookie, we should use exactly the same path and domain options as when we set it. +Please note: when we update or delete a cookie, we should use exactly the same path and domain attributes as when we set it. ``` Together: [cookie.js](cookie.js). @@ -349,11 +361,11 @@ Together: [cookie.js](cookie.js). ## Appendix: Third-party cookies -A cookie is called "third-party" if it's placed by domain other than the user is visiting. +A cookie is called "third-party" if it's placed by a domain other than the page the user is visiting. For instance: 1. A page at `site.com` loads a banner from another site: `<img src="https://ads.com/banner.png">`. -2. Along with the banner, the remote server at `ads.com` may set `Set-Cookie` header with cookie like `id=1234`. Such cookie originates from `ads.com` domain, and will only be visible at `ads.com`: +2. Along with the banner, the remote server at `ads.com` may set the `Set-Cookie` header with a cookie like `id=1234`. Such a cookie originates from the `ads.com` domain, and will only be visible at `ads.com`:  @@ -361,14 +373,14 @@ For instance:  -4. What's even more important, when the users moves from `site.com` to another site `other.com` that also has a banner, then `ads.com` gets the cookie, as it belongs to `ads.com`, thus recognizing the visitor and tracking him as he moves between sites: +4. What's even more important is, when the user moves from `site.com` to another site `other.com`, which also has a banner, then `ads.com` gets the cookie, as it belongs to `ads.com`, thus recognizing the visitor and tracking him as he moves between sites:  Third-party cookies are traditionally used for tracking and ads services, due to their nature. They are bound to the originating domain, so `ads.com` can track the same user between different sites, if they all access it. -Naturally, some people don't like being tracked, so browsers allow to disable such cookies. +Naturally, some people don't like being tracked, so browsers allow them to disable such cookies. Also, some modern browsers employ special policies for such cookies: - Safari does not allow third-party cookies at all. @@ -378,49 +390,49 @@ Also, some modern browsers employ special policies for such cookies: ```smart If we load a script from a third-party domain, like `<script src="https://google-analytics.com/analytics.js">`, and that script uses `document.cookie` to set a cookie, then such cookie is not third-party. -If a script sets a cookie, then no matter where the script came from -- it belongs to the domain of the current webpage. +If a script sets a cookie, then no matter where the script came from -- the cookie belongs to the domain of the current webpage. ``` ## Appendix: GDPR -This topic is not related to JavaScript at all, just something to keep in mind when setting cookies. +This topic is not related to JavaScript at all, it is just something to keep in mind when setting cookies. -There's a legislation in Europe called GDPR, that enforces a set of rules for websites to respect users' privacy. And one of such rules is to require an explicit permission for tracking cookies from a user. +There's a legislation in Europe called GDPR, that enforces a set of rules for websites to respect the users' privacy. One of these rules is to require explicit permission for tracking cookies from the user. -Please note, that's only about tracking/identifying cookies. +Please note, that's only about tracking/identifying/authorizing cookies. So, if we set a cookie that just saves some information, but neither tracks nor identifies the user, then we are free to do it. -But if we are going to set a cookie with an authentication session or a tracking id, then a user must allow that. +But if we are going to set a cookie with an authentication session or a tracking ID, then a user must allow that. -Websites generally have two variants of following GDPR. You must have seen them both already in the web: +Websites generally have two variants of complying with GDPR. You are likely to have seen them both on the web: 1. If a website wants to set tracking cookies only for authenticated users. - To do so, the registration form should have a checkbox like "accept the privacy policy", the user must check it, and then the website is free to set auth cookies. + To do so, the registration form should have a checkbox like "accept the privacy policy" (that describes how cookies are used), the user must check it, and then the website is free to set auth cookies. 2. If a website wants to set tracking cookies for everyone. - To do so legally, a website shows a modal "splash screen" for newcomers, and require them to agree for cookies. Then the website can set them and let people see the content. That can be disturbing for new visitors though. No one likes to see "must-click" modal splash screens instead of the content. But GDPR requires an explicit agreement. + To do so legally, a website shows a modal "splash screen" for newcomers and requires them to agree to the cookies. Then the website can set them and let people see the content. That can be disturbing for new visitors though. No one likes to see such "must-click" modal splash screens instead of the content. But GDPR requires an explicit agreement. -GDPR is not only about cookies, it's about other privacy-related issues too, but that's too much beyond our scope. +GDPR is not only about cookies, it is about other privacy-related issues too, but that is beyond our scope. ## Summary -`document.cookie` provides access to cookies -- write operations modify only cookies mentioned in it. -- name/value must be encoded. -- one cookie up to 4kb, 20+ cookies per site (depends on a browser). +`document.cookie` provides access to cookies. +- Write operations modify only the cookie mentioned in it. +- Name/value must be encoded. +- One cookie may not exceed 4KB in size. The number of cookies allowed on a domain is around 20+ (varies by browser). -Cookie options: +Cookie attributes: - `path=/`, by default current path, makes the cookie visible only under that path. -- `domain=site.com`, by default a cookie is visible on current domain only, if set explicitly to the domain, makes the cookie visible on subdomains. -- `expires` or `max-age` sets cookie expiration time, without them the cookie dies when the browser is closed. +- `domain=site.com`, by default a cookie is visible on the current domain only. If the domain is set explicitly, the cookie becomes visible on subdomains. +- `expires` or `max-age` sets the cookie expiration time. Without them, the cookie dies when the browser is closed. - `secure` makes the cookie HTTPS-only. -- `samesite` forbids the browser to send the cookie with requests coming from outside the site, helps to prevent XSRF attacks. +- `samesite` forbids the browser to send the cookie with requests coming from outside the site. This helps to prevent XSRF attacks. Additionally: -- Third-party cookies may be forbidden by the browser, e.g. Safari does that by default. +- The browser may forbid third-party cookies, e.g. Safari does that by default. There is also work in progress to implement this in Chrome. - When setting a tracking cookie for EU citizens, GDPR requires to ask for permission. diff --git a/6-data-storage/01-cookie/cookie.js b/6-data-storage/01-cookie/cookie.js index 7d8eebf3e..653ea5b7f 100644 --- a/6-data-storage/01-cookie/cookie.js +++ b/6-data-storage/01-cookie/cookie.js @@ -1,6 +1,6 @@ function getCookie(name) { let matches = document.cookie.match(new RegExp( - "(?:^|; )" + name.replace(/([\.$?*|{}\(\)\[\]\\\/\+^])/g, '\\$1') + "=([^;]*)" + "(?:^|; )" + name.replace(/([.$?*|{}()[\]\\/+^])/g, '\\$1') + "=([^;]*)" )); return matches ? decodeURIComponent(matches[1]) : undefined; } @@ -13,7 +13,7 @@ function setCookie(name, value, options = {}) { ...options }; - if (options.expires.toUTCString) { + if (options.expires instanceof Date) { options.expires = options.expires.toUTCString(); } diff --git a/6-data-storage/02-localstorage/1-form-autosave/task.md b/6-data-storage/02-localstorage/1-form-autosave/task.md index b2e0a4222..2b147d79c 100644 --- a/6-data-storage/02-localstorage/1-form-autosave/task.md +++ b/6-data-storage/02-localstorage/1-form-autosave/task.md @@ -3,7 +3,7 @@ Create a `textarea` field that "autosaves" its value on every change. -So, if the user occasionally closes the page, and opens it again, he'll find his unfinished input at place. +So, if the user accidentally closes the page, and opens it again, he'll find his unfinished input at place. Like this: diff --git a/6-data-storage/02-localstorage/article.md b/6-data-storage/02-localstorage/article.md index 128a9698e..a99bcb653 100644 --- a/6-data-storage/02-localstorage/article.md +++ b/6-data-storage/02-localstorage/article.md @@ -6,11 +6,11 @@ What's interesting about them is that the data survives a page refresh (for `ses We already have cookies. Why additional objects? -- Unlike cookies, web storage objects are not sent to server with each request. Because of that, we can store much more. Most browsers allow at least 2 megabytes of data (or more) and have settings to configure that. -- The server can't manipulate storage objects via HTTP headers, everything's done in JavaScript. +- Unlike cookies, web storage objects are not sent to server with each request. Because of that, we can store much more. Most modern browsers allow at least 5 megabytes of data (or more) and have settings to configure that. +- Also unlike cookies, the server can't manipulate storage objects via HTTP headers. Everything's done in JavaScript. - The storage is bound to the origin (domain/protocol/port triplet). That is, different protocols or subdomains infer different storage objects, they can't access data from each other. -Both storage objects provide same methods and properties: +Both storage objects provide the same methods and properties: - `setItem(key, value)` -- store key/value pair. - `getItem(key)` -- get the value by key. @@ -19,6 +19,8 @@ Both storage objects provide same methods and properties: - `key(index)` -- get the key on a given position. - `length` -- the number of stored items. +As you can see, it's like a `Map` collection (`setItem/getItem/removeItem`), but also allows access by index with `key(index)`. + Let's see how it works. ## localStorage demo @@ -40,9 +42,9 @@ localStorage.setItem('test', 1); alert( localStorage.getItem('test') ); // 1 ``` -We only have to be on the same domain/port/protocol, the url path can be different. +We only have to be on the same origin (domain/port/protocol), the url path can be different. -The `localStorage` is shared, so if we set the data in one window, the change becomes visible in the other one. +The `localStorage` is shared between all windows with the same origin, so if we set the data in one window, the change becomes visible in another one. ## Object-like access @@ -59,9 +61,10 @@ alert( localStorage.test ); // 2 delete localStorage.test; ``` -That's allowed for historical reasons, and mostly works, but generally not recommended for two reasons: +That's allowed for historical reasons, and mostly works, but generally not recommended, because: 1. If the key is user-generated, it can be anything, like `length` or `toString`, or another built-in method of `localStorage`. In that case `getItem/setItem` work fine, while object-like access fails: + ```js run let key = 'length'; localStorage[key] = 5; // Error, can't assign length @@ -71,11 +74,11 @@ That's allowed for historical reasons, and mostly works, but generally not recom ## Looping over keys -Methods provide get/set/remove functionality. But how to get all the keys? +As we've seen, the methods provide "get/set/remove by key" functionality. But how to get all saved values or keys? Unfortunately, storage objects are not iterable. -One way is to use "array-like" iteration: +One way is to loop over them as over an array: ```js run for(let i=0; i<localStorage.length; i++) { @@ -84,9 +87,9 @@ for(let i=0; i<localStorage.length; i++) { } ``` -Another way is to use object-specific `for key in localStorage` loop. +Another way is to use `for key in localStorage` loop, just as we do with regular objects. -That iterates over keys, but also outputs few built-in fields that we don't need: +It iterates over keys, but also outputs few built-in fields that we don't need: ```js run // bad try @@ -117,25 +120,24 @@ for(let key of keys) { The latter works, because `Object.keys` only returns the keys that belong to the object, ignoring the prototype. - ## Strings only Please note that both key and value must be strings. -If we any other type, like a number, or an object, it gets converted to string automatically: +If they were any other type, like a number, or an object, they would get converted to a string automatically: ```js run -sessionStorage.user = {name: "John"}; -alert(sessionStorage.user); // [object Object] +localStorage.user = {name: "John"}; +alert(localStorage.user); // [object Object] ``` We can use `JSON` to store objects though: ```js run -sessionStorage.user = JSON.stringify({name: "John"}); +localStorage.user = JSON.stringify({name: "John"}); // sometime later -let user = JSON.parse( sessionStorage.user ); +let user = JSON.parse( localStorage.user ); alert( user.name ); // John ``` @@ -146,7 +148,6 @@ Also it is possible to stringify the whole storage object, e.g. for debugging pu alert( JSON.stringify(localStorage, null, 2) ); ``` - ## sessionStorage The `sessionStorage` object is used much less often than `localStorage`. @@ -155,7 +156,7 @@ Properties and methods are the same, but it's much more limited: - The `sessionStorage` exists only within the current browser tab. - Another tab with the same page will have a different storage. - - But it is shared between iframes in the tab (assuming they come from the same origin). + - But it is shared between iframes in the same tab (assuming they come from the same origin). - The data survives page refresh, but not closing/opening the tab. Let's see that in action. @@ -178,7 +179,7 @@ That's exactly because `sessionStorage` is bound not only to the origin, but als ## Storage event -When the data gets updated in `localStorage` or `sessionStorage`, [storage](https://www.w3.org/TR/webstorage/#the-storage-event) event triggers, with properties: +When the data gets updated in `localStorage` or `sessionStorage`, [storage](https://html.spec.whatwg.org/multipage/webstorage.html#the-storageevent-interface) event triggers, with properties: - `key` – the key that was changed (`null` if `.clear()` is called). - `oldValue` – the old value (`null` if the key is newly added). @@ -196,11 +197,11 @@ Imagine, you have two windows with the same site in each. So `localStorage` is s You might want to open this page in two browser windows to test the code below. ``` -Now if both windows are listening for `window.onstorage`, then each one will react on updates that happened in the other one. +If both windows are listening for `window.onstorage`, then each one will react on updates that happened in the other one. ```js run // triggers on updates made to the same storage from other documents -window.onstorage = event => { +window.onstorage = event => { // can also use window.addEventListener('storage', event => { if (event.key != 'now') return; alert(event.key + ':' + event.newValue + " at " + event.url); }; @@ -210,24 +211,25 @@ localStorage.setItem('now', Date.now()); Please note that the event also contains: `event.url` -- the url of the document where the data was updated. -Also, `event.storageArea` contains the storage object -- the event is the same for both `sessionStorage` and `localStorage`, so `storageArea` references the one that was modified. We may event want to set something back in it, to "respond" to a change. +Also, `event.storageArea` contains the storage object -- the event is the same for both `sessionStorage` and `localStorage`, so `event.storageArea` references the one that was modified. We may even want to set something back in it, to "respond" to a change. **That allows different windows from the same origin to exchange messages.** -Modern browsers also support [Broadcast channel API](https://developer.mozilla.org/en-US/docs/Web/API/Broadcast_Channel_API), the special API for same-origin inter-window communication, it's more full featured, but less supported. There are libraries that polyfill that API, based on `localStorage`, that make it available everywhere. +Modern browsers also support [Broadcast channel API](mdn:/api/Broadcast_Channel_API), the special API for same-origin inter-window communication, it's more full featured, but less supported. There are libraries that polyfill that API, based on `localStorage`, that make it available everywhere. ## Summary -Web storage objects `localStorage` and `sessionStorage` allow to store key/value in the browser. +Web storage objects `localStorage` and `sessionStorage` allow to store key/value pairs in the browser. + - Both `key` and `value` must be strings. -- The limit is 2mb+, depends on the browser. +- The limit is 5mb+, depends on the browser. - They do not expire. - The data is bound to the origin (domain/port/protocol). | `localStorage` | `sessionStorage` | |----------------|------------------| | Shared between all tabs and windows with the same origin | Visible within a browser tab, including iframes from the same origin | -| Survives browser restart | Dies on tab close | +| Survives browser restart | Survives page refresh (but not tab close) | API: @@ -235,13 +237,13 @@ API: - `getItem(key)` -- get the value by key. - `removeItem(key)` -- remove the key with its value. - `clear()` -- delete everything. -- `key(index)` -- get the key on a given position. +- `key(index)` -- get the key number `index`. - `length` -- the number of stored items. - Use `Object.keys` to get all keys. -- Can use the keys as object properties, in that case `storage` event doesn't trigger. +- We access keys as object properties, in that case `storage` event isn't triggered. Storage event: - Triggers on `setItem`, `removeItem`, `clear` calls. -- Contains all the data about the operation, the document `url` and the storage object. +- Contains all the data about the operation (`key/oldValue/newValue`), the document `url` and the storage object `storageArea`. - Triggers on all `window` objects that have access to the storage except the one that generated it (within a tab for `sessionStorage`, globally for `localStorage`). diff --git a/6-data-storage/03-indexeddb/article.md b/6-data-storage/03-indexeddb/article.md index 5218ce850..43344e487 100644 --- a/6-data-storage/03-indexeddb/article.md +++ b/6-data-storage/03-indexeddb/article.md @@ -5,22 +5,28 @@ libs: # IndexedDB -IndexedDB is a built-in database, much more powerful than `localStorage`. +IndexedDB is a database that is built into a browser, much more powerful than `localStorage`. -- Key/value storage: value can be (almost) anything, multiple key types. +- Stores almost any kind of values by keys, multiple key types. - Supports transactions for reliability. - Supports key range queries, indexes. -- Can store much more data than `localStorage`. +- Can store much bigger volumes of data than `localStorage`. That power is usually excessive for traditional client-server apps. IndexedDB is intended for offline apps, to be combined with ServiceWorkers and other technologies. The native interface to IndexedDB, described in the specification <https://www.w3.org/TR/IndexedDB>, is event-based. -We can also use `async/await` with the help of a promise-based wrapper, like <https://github.com/jakearchibald/idb>. That's pretty convenient, but the wrapper is not perfect, it can't replace events for all cases, so we'll start with events, and then use the wrapper. +We can also use `async/await` with the help of a promise-based wrapper, like <https://github.com/jakearchibald/idb>. That's pretty convenient, but the wrapper is not perfect, it can't replace events for all cases. So we'll start with events, and then, after we gain an understanding of IndexedDB, we'll use the wrapper. + +```smart header="Where's the data?" +Technically, the data is usually stored in the visitor's home directory, along with browser settings, extensions, etc. + +Different browsers and OS-level users have each their own independant storage. +``` ## Open database -To start working with IndexedDB, we need to open a database. +To start working with IndexedDB, we first need to `open` (connect to) a database. The syntax: @@ -31,22 +37,24 @@ let openRequest = indexedDB.open(name, version); - `name` -- a string, the database name. - `version` -- a positive integer version, by default `1` (explained below). -We can have many databases with different names, all within the current origin (domain/protocol/port). So different websites can't access databases of each other. +We can have many databases with different names, but all of them exist within the current origin (domain/protocol/port). Different websites can't access each other's databases. -After the call, we need to listen to events on `openRequest` object: -- `success`: database is ready, use the database object `openRequest.result` for further work. -- `error`: open failed. -- `upgradeneeded`: database version is outdated (see below). +The call returns `openRequest` object, we should listen to events on it: +- `success`: database is ready, there's the "database object" in `openRequest.result`, we should use it for further calls. +- `error`: opening failed. +- `upgradeneeded`: database is ready, but its version is outdated (see below). **IndexedDB has a built-in mechanism of "schema versioning", absent in server-side databases.** -Unlike server-side databases, IndexedDB is client-side, we don't have the data at hands. But when we publish a new version of our app, we may need to update the database. +Unlike server-side databases, IndexedDB is client-side, the data is stored in the browser, so we, developers, don't have full-time access to it. So, when we have published a new version of our app, and the user visits our webpage, we may need to update the database. If the local database version is less than specified in `open`, then a special event `upgradeneeded` is triggered, and we can compare versions and upgrade data structures as needed. -The event also triggers when the database did not exist yet, so we can perform initialization. +The `upgradeneeded` event also triggers when the database doesn't yet exist (technically, its version is `0`), so we can perform the initialization. + +Let's say we published the first version of our app. -For instance, when we first publish our app, we open it with version `1` and perform the initialization in `upgradeneeded` handler: +Then we can open the database with version `1` and perform the initialization in an `upgradeneeded` handler like this: ```js let openRequest = indexedDB.open("store", *!*1*/!*); @@ -57,24 +65,26 @@ openRequest.onupgradeneeded = function() { }; openRequest.onerror = function() { - console.error("Error", openResult.error); + console.error("Error", openRequest.error); }; openRequest.onsuccess = function() { let db = openRequest.result; - // continue to work with database using db object + // continue working with database using db object }; ``` -When we publish the 2nd version: +Then, later, we publish the 2nd version. + +We can open it with version `2` and perform the upgrade like this: ```js let openRequest = indexedDB.open("store", *!*2*/!*); -// check the existing database version, do the updates if needed: -openRequest.onupgradeneeded = function() { +openRequest.onupgradeneeded = function(event) { + // the existing database version is less than 2 (or it doesn't exist) let db = openRequest.result; - switch(db.version) { // existing (old) db version + switch(event.oldVersion) { // existing db version case 0: // version 0 means that the client had no database // perform initialization @@ -85,7 +95,9 @@ openRequest.onupgradeneeded = function() { }; ``` -After `openRequest.onsuccess` we have the database object in `openRequest.result`, that we'll use for further operations. +Please note: as our current version is `2`, the `onupgradeneeded` handler has a code branch for version `0`, suitable for users that are accessing for the first time and have no database, and also for version `1`, for upgrades. + +And then, only if `onupgradeneeded` handler finishes without errors, `openRequest.onsuccess` triggers, and the database is considered successfully opened. To delete a database: @@ -94,9 +106,77 @@ let deleteRequest = indexedDB.deleteDatabase(name) // deleteRequest.onsuccess/onerror tracks the result ``` +```warn header="We can't open a database using an older open call version" +If the current user database has a higher version than in the `open` call, e.g. the existing DB version is `3`, and we try to `open(...2)`, then that's an error, `openRequest.onerror` triggers. + +That's rare, but such a thing may happen when a visitor loads outdated JavaScript code, e.g. from a proxy cache. So the code is old, but his database is new. + +To protect from errors, we should check `db.version` and suggest a page reload. Use proper HTTP caching headers to avoid loading the old code, so that you'll never have such problems. +``` + +### Parallel update problem + +As we're talking about versioning, let's tackle a small related problem. + +Let's say: +1. A visitor opened our site in a browser tab, with database version `1`. +2. Then we rolled out an update, so our code is newer. +3. And then the same visitor opens our site in another tab. + +So there's a tab with an open connection to DB version `1`, while the second one attempts to update it to version `2` in its `upgradeneeded` handler. + +The problem is that a database is shared between two tabs, as it's the same site, same origin. And it can't be both version `1` and `2`. To perform the update to version `2`, all connections to version 1 must be closed, including the one in the first tab. + +In order to organize that, the `versionchange` event triggers on the "outdated" database object. We should listen for it and close the old database connection (and probably suggest a page reload, to load the updated code). + +If we don't listen for the `versionchange` event and don't close the old connection, then the second, new connection won't be made. The `openRequest` object will emit the `blocked` event instead of `success`. So the second tab won't work. + +Here's the code to correctly handle the parallel upgrade. It installs the `onversionchange` handler, that triggers if the current database connection becomes outdated (db version is updated elsewhere) and closes the connection. + +```js +let openRequest = indexedDB.open("store", 2); + +openRequest.onupgradeneeded = ...; +openRequest.onerror = ...; + +openRequest.onsuccess = function() { + let db = openRequest.result; + + *!* + db.onversionchange = function() { + db.close(); + alert("Database is outdated, please reload the page.") + }; + */!* + + // ...the db is ready, use it... +}; + +*!* +openRequest.onblocked = function() { + // this event shouldn't trigger if we handle onversionchange correctly + + // it means that there's another open connection to the same database + // and it wasn't closed after db.onversionchange triggered for it +}; +*/!* +``` + +...In other words, here we do two things: + +1. The `db.onversionchange` listener informs us about a parallel update attempt, if the current database version becomes outdated. +2. The `openRequest.onblocked` listener informs us about the opposite situation: there's a connection to an outdated version elsewhere, and it doesn't close, so the newer connection can't be made. + +We can handle things more gracefully in `db.onversionchange`, prompt the visitor to save the data before the connection is closed and so on. + +Or, an alternative approach would be to not close the database in `db.onversionchange`, but instead use the `onblocked` handler (in the new tab) to alert the visitor, tell him that the newer version can't be loaded until they close other tabs. + +These update collisions happen rarely, but we should at least have some handling for them, at least an `onblocked` handler, to prevent our script from dying silently. ## Object store +To store something in IndexedDB, we need an *object store*. + An object store is a core concept of IndexedDB. Counterparts in other databases are called "tables" or "collections". It's where the data is stored. A database may have multiple stores: one for users, another one for goods, etc. Despite being named an "object store", primitives can be stored too. @@ -105,17 +185,20 @@ Despite being named an "object store", primitives can be stored too. IndexedDB uses the [standard serialization algorithm](https://www.w3.org/TR/html53/infrastructure.html#section-structuredserializeforstorage) to clone-and-store an object. It's like `JSON.stringify`, but more powerful, capable of storing much more datatypes. -An example of object that can't be stored: an object with circular references. Such objects are not serializable. `JSON.stringify` also fails for such objects. +An example of an object that can't be stored: an object with circular references. Such objects are not serializable. `JSON.stringify` also fails for such objects. -**There must be an unique `key` for every value in the store.** +**There must be a unique `key` for every value in the store.** -A key must have a type one of: number, date, string, binary, or array. It's a unique object identifier: we can search/remove/update values by the key. +A key must be one of these types - number, date, string, binary, or array. It's a unique identifier, so we can search/remove/update values by the key.  -We can provide a key when we add an value to the store, similar to `localStorage`. That's good for storing primitive values. But when we store objects, IndexedDB allows to setup an object property as the key, that's much more convenient. Or we can auto-generate keys. +As we'll see very soon, we can provide a key when we add a value to the store, similar to `localStorage`. But when we store objects, IndexedDB allows setting up an object property as the key, which is much more convenient. Or we can auto-generate keys. + +But we need to create an object store first. The syntax to create an object store: + ```js db.createObjectStore(name[, keyOptions]); ``` @@ -127,36 +210,39 @@ Please note, the operation is synchronous, no `await` needed. - `keyPath` -- a path to an object property that IndexedDB will use as the key, e.g. `id`. - `autoIncrement` -- if `true`, then the key for a newly stored object is generated automatically, as an ever-incrementing number. -If we don't supply any options, then we'll need to provide a key explicitly later, when storing an object. +If we don't supply `keyOptions`, then we'll need to provide a key explicitly later, when storing an object. For instance, this object store uses `id` property as the key: + ```js db.createObjectStore('books', {keyPath: 'id'}); ``` **An object store can only be created/modified while updating the DB version, in `upgradeneeded` handler.** -That's a technical limitation. Outside of the handler we'll be able to add/remove/update the data, but object stores are changed only during version update. +That's a technical limitation. Outside of the handler we'll be able to add/remove/update the data, but object stores can only be created/removed/altered during a version update. + +To perform a database version upgrade, there are two main approaches: -To do an upgrade, there are two main ways: -1. We can compare versions and run per-version operations. -2. Or we can get a list of existing object stores as `db.objectStoreNames`. That object is a [DOMStringList](https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#domstringlist), and it provides `contains(name)` method to check for the existance. And then we can do updates depending on what exists. +1. We can implement per-version upgrade functions: from 1 to 2, from 2 to 3, from 3 to 4 etc. Then, in `upgradeneeded` we can compare versions (e.g. old 2, now 4) and run per-version upgrades step by step, for every intermediate version (2 to 3, then 3 to 4). +2. Or we can just examine the database: get a list of existing object stores as `db.objectStoreNames`. That object is a [DOMStringList](https://html.spec.whatwg.org/multipage/common-dom-interfaces.html#domstringlist) that provides `contains(name)` method to check for existance. And then we can do updates depending on what exists and what doesn't. + +For small databases the second variant may be simpler. Here's the demo of the second approach: ```js -let openRequest = indexedDB.open("db", 1); +let openRequest = indexedDB.open("db", 2); -// create an object store for books if not exists +// create/upgrade the database without version checks openRequest.onupgradeneeded = function() { let db = openRequest.result; - if (!db.objectStoreNames.contains('books')) { - db.createObjectStore('books', {keyPath: 'id'}); + if (!db.objectStoreNames.contains('books')) { // if there's no "books" store + db.createObjectStore('books', {keyPath: 'id'}); // create it } }; ``` - To delete an object store: ```js @@ -167,9 +253,10 @@ db.deleteObjectStore('books') The term "transaction" is generic, used in many kinds of databases. -A transaction is a group operations, that should either all succeed or all fail. +A transaction is a group of operations, that should either all succeed or all fail. + +For instance, when a person buys something, we need to: -For instance, when a person buys something, we need: 1. Subtract the money from their account. 2. Add the item to their inventory. @@ -181,21 +268,21 @@ Transactions can guarantee that. To start a transaction: -```js run +```js db.transaction(store[, type]); ``` - `store` is a store name that the transaction is going to access, e.g. `"books"`. Can be an array of store names if we're going to access multiple stores. - `type` – a transaction type, one of: - `readonly` -- can only read, the default. - - `readwrite` -- can only read and write, but not modify object stores. + - `readwrite` -- can only read and write the data, but not create/remove/alter object stores. -There'is also `versionchange` transaction type: such transactions can do everything, but we can't create them manually. IndexedDB automatically creates a `versionchange` transaction when opening the database, for `updateneeded` handler. That's why it's a single place where we can update the database structure, create/remove object stores. +There's also `versionchange` transaction type: such transactions can do everything, but we can't create them manually. IndexedDB automatically creates a `versionchange` transaction when opening the database, for `upgradeneeded` handler. That's why it's a single place where we can update the database structure, create/remove object stores. -```smart header="What are transaction types for?" +```smart header="Why are there different types of transactions?" Performance is the reason why transactions need to be labeled either `readonly` and `readwrite`. -Many `readonly` transactions can access concurrently the same store, but `readwrite` transactions can't. A `readwrite` transaction "locks" the store for writing. The next transaction must wait before the previous one finishes before accessing the same store. +Many `readonly` transactions are able to access the same store concurrently, but `readwrite` transactions can't. A `readwrite` transaction "locks" the store for writing. The next transaction must wait before the previous one finishes before accessing the same store. ``` After the transaction is created, we can add an item to the store, like this: @@ -227,29 +314,29 @@ request.onerror = function() { }; ``` -There are basically four steps: +There were basically four steps: -1. Create a transaction, mention all stores it's going to access, at `(1)`. +1. Create a transaction, mentioning all the stores it's going to access, at `(1)`. 2. Get the store object using `transaction.objectStore(name)`, at `(2)`. 3. Perform the request to the object store `books.add(book)`, at `(3)`. -4. ...Handle request success/error `(4)`, make other requests if needed, etc. +4. ...Handle request success/error `(4)`, then we can make other requests if needed, etc. Object stores support two methods to store a value: - **put(value, [key])** - Add the `value` to the store. The `key` is supplied only if the object store did not have `keyPath` or `autoIncrement` option. If there's already a value with same key, it will be replaced. + Add the `value` to the store. The `key` is supplied only if the object store did not have `keyPath` or `autoIncrement` option. If there's already a value with the same key, it will be replaced. - **add(value, [key])** Same as `put`, but if there's already a value with the same key, then the request fails, and an error with the name `"ConstraintError"` is generated. -Just like when opening a database, we send a request: `books.add(book)`, and then wait for `success/error` events. +Similar to opening a database, we can send a request: `books.add(book)`, and then wait for `success/error` events. - The `request.result` for `add` is the key of the new object. - The error is in `request.error` (if any). -## Transactions autocommit +## Transactions' autocommit -In the example above we started the transaction and made `add` request. We could make more requests. How do we finish ("commit") the transaction? +In the example above we started the transaction and made `add` request. But as we stated previously, a transaction may have multiple associated requests, that must either all succeed or all fail. How do we mark the transaction as finished, with no more requests to come? The short answer is: we don't. @@ -257,19 +344,13 @@ In the next version 3.0 of the specification, there will probably be a manual wa **When all transaction requests are finished, and the [microtasks queue](info:microtask-queue) is empty, it is committed automatically.** -```smart header="What's an \"empty microtask queue\"?" -The microtask queue is explained in [another chapter](info:async-await#microtask-queue). In short, an empty microtask queue means that for all settled promises their `.then/catch/finally` handlers are executed. - -In other words, handling of finished promises and resuming "awaits" is done before closing the transaction. - -That's a minor technical detail. If we're using `async/await` instead of low-level promise calls, then we can assume that a transaction commits when all its requests are done, and the current code finishes. -``` +Usually, we can assume that a transaction commits when all its requests are complete, and the current code finishes. -So, in the example above no special code is needed to finish the transaction. +So, in the example above no special call is needed to finish the transaction. -Transactions auto-commit principle has an important side effect. We can't insert an async operation like `fetch`, `setTimeout` in the middle of transaction. IndexedDB will not keep the transaction waiting till these are done. +Transactions auto-commit principle has an important side effect. We can't insert an async operation like `fetch`, `setTimeout` in the middle of a transaction. IndexedDB will not keep the transaction waiting till these are done. -In the code below `request2` in line `(*)` fails, because the transaction is already committed, can't make any request in it: +In the code below, `request2` in the line `(*)` fails, because the transaction is already committed, and can't make any request in it: ```js let request1 = books.add(book); @@ -290,7 +371,7 @@ That's because `fetch` is an asynchronous operation, a macrotask. Transactions a Authors of IndexedDB spec believe that transactions should be short-lived. Mostly for performance reasons. -Notably, `readwrite` transactions "lock" the stores for writing. So if one part of application initiated `readwrite` on `books` object store, then another part that wants to do the same has to wait: the new transaction "hangs" till the first one is done. That can lead to strange delays if transactions take a long time. +Notably, `readwrite` transactions "lock" the stores for writing. So if one part of the application initiated `readwrite` on `books` object store, then another part that wants to do the same has to wait: the new transaction "hangs" till the first one is done. That can lead to strange delays if transactions take a long time. So, what to do? @@ -331,9 +412,9 @@ That's to be expected, not only because of possible errors at our side, but also **A failed request automatically aborts the transaction, canceling all its changes.** -Sometimes a request may fail with a non-critical error. We'd like to handle it in `request.onerror` and continue the transaction. Then, to prevent the transaction abort, we should call `event.preventDefault()`. +In some situations, we may want to handle the failure (e.g. try another request), without canceling existing changes, and continue the transaction. That's possible. The `request.onerror` handler is able to prevent the transaction abort by calling `event.preventDefault()`. -In the example below a new book is added with the same key (`id`). The `store.add` method generates a `"ConstraintError"` in that case. We handle it without canceling the transaction: +In the example below a new book is added with the same key (`id`) as the existing one. The `store.add` method generates a `"ConstraintError"` in that case. We handle it without canceling the transaction: ```js let transaction = db.transaction("books", "readwrite"); @@ -347,6 +428,7 @@ request.onerror = function(event) { if (request.error.name == "ConstraintError") { console.log("Book with such id already exists"); // handle the error event.preventDefault(); // don't abort the transaction + // use another key for the book? } else { // unexpected error, can't handle it // the transaction will abort @@ -394,24 +476,29 @@ request.onerror = function(event) { }; ``` -## Searching by keys +## Searching + +There are two main types of search in an object store: -There are two main ways to search in an object store: -1. By a key or a key range. That is: by `book.id` in our "books" storage. -2. By another object field, e.g. `book.price`. We need an index for that. +1. By a key value or a key range. In our "books" storage that would be a value or range of values of `book.id`. +2. By another object field, e.g. `book.price`. This required an additional data structure, named "index". -First let's deal with the keys and key ranges `(1)`. +### By key -Methods that involve searching support either exact keys or so-called "range queries" -- [IDBKeyRange](https://www.w3.org/TR/IndexedDB/#keyrange) objects that specify a "key range". +First let's deal with the first type of search: by key. -Ranges are created using following calls: +Searching methods support both exact key values and so-called "ranges of values" -- [IDBKeyRange](https://www.w3.org/TR/IndexedDB/#keyrange) objects that specify an acceptable "key range". -- `IDBKeyRange.lowerBound(lower, [open])` means: `>lower` (or `≥lower` if `open` is true) -- `IDBKeyRange.upperBound(upper, [open])` means: `<upper` (or `≤upper` if `open` is true) -- `IDBKeyRange.bound(lower, upper, [lowerOpen], [upperOpen])` means: between `lower` and `upper`, with optional equality if the corresponding `open` is true. +`IDBKeyRange` objects are created using following calls: + +- `IDBKeyRange.lowerBound(lower, [open])` means: `≥lower` (or `>lower` if `open` is true) +- `IDBKeyRange.upperBound(upper, [open])` means: `≤upper` (or `<upper` if `open` is true) +- `IDBKeyRange.bound(lower, upper, [lowerOpen], [upperOpen])` means: between `lower` and `upper`. If the open flags is true, the corresponding key is not included in the range. - `IDBKeyRange.only(key)` -- a range that consists of only one `key`, rarely used. -All searching methods accept a `query` argument that can be either an exact key or a key range: +We'll see practical examples of using them very soon. + +To perform the actual search, there are following methods. They accept a `query` argument that can be either an exact key or a key range: - `store.get(query)` -- search for the first value by a key or a range. - `store.getAll([query], [count])` -- search for all values, limit by `count` if given. @@ -427,27 +514,26 @@ Request examples: // get one book books.get('js') -// get books with 'css' < id < 'html' +// get books with 'css' <= id <= 'html' books.getAll(IDBKeyRange.bound('css', 'html')) -// get books with 'html' <= id -books.getAll(IDBKeyRange.lowerBound('html', true)) +// get books with id < 'html' +books.getAll(IDBKeyRange.upperBound('html', true)) // get all books books.getAll() -// get all keys: id >= 'js' +// get all keys, where id > 'js' books.getAllKeys(IDBKeyRange.lowerBound('js', true)) ``` ```smart header="Object store is always sorted" -Object store sorts values by key internally. +An object store sorts values by key internally. So requests that return many values always return them in sorted by key order. ``` - -## Searching by any field with an index +### By a field using an index To search by other object fields, we need to create an additional data structure named "index". @@ -463,7 +549,7 @@ objectStore.createIndex(name, keyPath, [options]); - **`keyPath`** -- path to the object field that the index should track (we're going to search by that field), - **`option`** -- an optional object with properties: - **`unique`** -- if true, then there may be only one object in the store with the given value at the `keyPath`. The index will enforce that by generating an error if we try to add a duplicate. - - **`multiEntry`** -- only used if there value on `keyPath` is an array. In that case, by default, the index will treat the whole array as the key. But if `multiEntry` is true, then the index will keep a list of store objects for each value in that array. So array members become index keys. + - **`multiEntry`** -- only used if the value on `keyPath` is an array. In that case, by default, the index will treat the whole array as the key. But if `multiEntry` is true, then the index will keep a list of store objects for each value in that array. So array members become index keys. In our example, we store books keyed by `id`. @@ -476,7 +562,7 @@ openRequest.onupgradeneeded = function() { // we must create the index here, in versionchange transaction let books = db.createObjectStore('books', {keyPath: 'id'}); *!* - let index = inventory.createIndex('price_idx', 'price'); + let index = books.createIndex('price_idx', 'price'); */!* }; ``` @@ -516,7 +602,7 @@ request.onsuccess = function() { We can also use `IDBKeyRange` to create ranges and looks for cheap/expensive books: ```js -// find books where price < 5 +// find books where price <= 5 let request = priceIndex.getAll(IDBKeyRange.upperBound(5)); ``` @@ -524,11 +610,12 @@ Indexes are internally sorted by the tracked object field, `price` in our case. ## Deleting from store -The `delete` method looks up values to delete by a query, just like `getAll`. +The `delete` method looks up values to delete by a query, the call format is similar to `getAll`: - **`delete(query)`** -- delete matching values by query. For instance: + ```js // delete the book with id='js' books.delete('js'); @@ -547,6 +634,7 @@ request.onsuccess = function() { ``` To delete everything: + ```js books.clear(); // clear the storage. ``` @@ -555,9 +643,7 @@ books.clear(); // clear the storage. Methods like `getAll/getAllKeys` return an array of keys/values. -But an object storage can be huge, bigger than the available memory. - -Then `getAll` will fail to get all records as an array. +But an object storage can be huge, bigger than the available memory. Then `getAll` will fail to get all records as an array. What to do? @@ -568,6 +654,7 @@ Cursors provide the means to work around that. As an object store is sorted internally by key, a cursor walks the store in key order (ascending by default). The syntax: + ```js // like getAll, but with a cursor: let request = store.openCursor(query, [direction]); @@ -608,15 +695,15 @@ request.onsuccess = function() { The main cursor methods are: - `advance(count)` -- advance the cursor `count` times, skipping values. -- `continue([key])` -- advance the cursor to the next value in range matching or after key. +- `continue([key])` -- advance the cursor to the next value in range matching (or immediately after `key` if given). Whether there are more values matching the cursor or not -- `onsuccess` gets called, and then in `result` we can get the cursor pointing to the next record, or `undefined`. In the example above the cursor was made for the object store. -But we also can make a cursor over an index. As we remember, indexes allow to search by an object field. Cursors over indexes to precisely the same as over object stores -- they save memory by returning one value at a timee. +But we also can make a cursor over an index. As we remember, indexes allow to search by an object field. Cursors over indexes do precisely the same as over object stores -- they save memory by returning one value at a time. -For cursors over indexes, `cursor.key` is the index key (e.g. price), and we should use `cursor.primaryKey` property the object key: +For cursors over indexes, `cursor.key` is the index key (e.g. price), and we should use `cursor.primaryKey` property for the object key: ```js let request = priceIdx.openCursor(IDBKeyRange.upperBound(5)); @@ -625,7 +712,7 @@ let request = priceIdx.openCursor(IDBKeyRange.upperBound(5)); request.onsuccess = function() { let cursor = request.result; if (cursor) { - let key = cursor.primaryKey; // next object store key (id field) + let primaryKey = cursor.primaryKey; // next object store key (id field) let value = cursor.value; // next object store object (book object) let key = cursor.key; // next index key (price) console.log(key, value); @@ -645,7 +732,7 @@ Let's use a thin promise wrapper <https://github.com/jakearchibald/idb> further Then, instead of `onsuccess/onerror` we can write like this: ```js -let db = await idb.openDb('store', 1, db => { +let db = await idb.openDB('store', 1, db => { if (db.oldVersion == 0) { // perform the initialization db.createObjectStore('books', {keyPath: 'id'}); @@ -665,14 +752,13 @@ try { } catch(err) { console.log('error', err.message); } - ``` So we have all the sweet "plain async code" and "try..catch" stuff. ### Error handling -If we don't catch the error, then it falls through, just as usual. +If we don't catch an error, then it falls through, till the closest outer `try..catch`. An uncaught error becomes an "unhandled promise rejection" event on `window` object. @@ -688,7 +774,7 @@ window.addEventListener('unhandledrejection', event => { ### "Inactive transaction" pitfall -A we know already, a transaction auto-commits as soon as the browser is done with the current code and microtasks. So if we put an *macrotask* like `fetch` in the middle of a transaction, then the transaction won't wait for it to finish. It just auto-commits. So the next request in it fails. +As we already know, a transaction auto-commits as soon as the browser is done with the current code and microtasks. So if we put a *macrotask* like `fetch` in the middle of a transaction, then the transaction won't wait for it to finish. It just auto-commits. So the next request in it would fail. For a promise wrapper and `async/await` the situation is the same. @@ -707,7 +793,8 @@ await inventory.add({ id: 'js', price: 10, created: new Date() }); // Error The next `inventory.add` after `fetch` `(*)` fails with an "inactive transaction" error, because the transaction is already committed and closed at that time. -The workaround is same as when working with native IndexedDB: either make a new transaction or just split things apart. +The workaround is the same as when working with native IndexedDB: either make a new transaction or just split things apart. + 1. Prepare the data and fetch all that's needed first. 2. Then save in the database. @@ -715,7 +802,7 @@ The workaround is same as when working with native IndexedDB: either make a new Internally, the wrapper performs a native IndexedDB request, adding `onerror/onsuccess` to it, and returns a promise that rejects/resolves with the result. -That works most fine of the time. The examples are at the lib page <https://github.com/jakearchibald/idb>. +That works fine most of the time. The examples are at the lib page <https://github.com/jakearchibald/idb>. In few rare cases, when we need the original `request` object, we can access it as `promise.request` property of the promise: @@ -734,14 +821,13 @@ let result = await promise; // if still needed IndexedDB can be thought of as a "localStorage on steroids". It's a simple key-value database, powerful enough for offline apps, yet simple to use. -The best manual is the specification, [the current one](https://w3c.github.io/IndexedDB) is 2.0, but few methods from [3.0](https://w3c.github.io/IndexedDB/) (it's not much different) are partially supported. +The best manual is the specification, [the current one](https://www.w3.org/TR/IndexedDB-2/) is 2.0, but few methods from [3.0](https://w3c.github.io/IndexedDB/) (it's not much different) are partially supported. -The usage can be described with a few phrases: +The basic usage can be described with a few phrases: 1. Get a promise wrapper like [idb](https://github.com/jakearchibald/idb). 2. Open a database: `idb.openDb(name, version, onupgradeneeded)` - - Create object storages in indexes in `onupgradeneeded` handlers. - - Update version if needed - either by comparing numbers or just checking what exists. + - Create object storages and indexes in `onupgradeneeded` handler or perform version update if needed. 3. For requests: - Create transaction `db.transaction('books')` (readwrite if needed). - Get the object store `transaction.objectStore('books')`. diff --git a/6-data-storage/03-indexeddb/indexeddb-cursor.svg b/6-data-storage/03-indexeddb/indexeddb-cursor.svg new file mode 100644 index 000000000..c41ac75d9 --- /dev/null +++ b/6-data-storage/03-indexeddb/indexeddb-cursor.svg @@ -0,0 +1,45 @@ +<?xml version="1.0" encoding="UTF-8"?> +<svg width="312px" height="291px" viewBox="0 0 312 291" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> + <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> + <title>indexeddb-cursor.svg</title> + <desc>Created with sketchtool.</desc> + <g id="data-storage" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> + <g id="indexeddb-cursor.svg"> + <path d="M161,43 L161,101 L268,101 L268,43 L161,43 Z" id="Rectangle-1" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB"></path> + <path d="M161,43 L161,101 L274,101 L274,43 L161,43 Z" id="Rectangle-1-Copy" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB"></path> + <text id="id:-'html'-price:-3" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="171" y="66">id: 'html'</tspan> + <tspan x="171" y="81">price: 3</tspan> + </text> + <path d="M161,101 L161,159 L274,159 L274,101 L161,101 Z" id="Rectangle-1-Copy-2" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB"></path> + <text id="id:-'css'-price:-5" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="171" y="124">id: 'css'</tspan> + <tspan x="171" y="139">price: 5</tspan> + </text> + <path d="M161,159 L161,217 L274,217 L274,159 L161,159 Z" id="Rectangle-1-Copy-3" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB"></path> + <text id="id:-'js'-price:-10" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="171" y="182">id: 'js'</tspan> + <tspan x="171" y="197">price: 10</tspan> + </text> + <path d="M161,217 L161,275 L274,275 L274,217 L161,217 Z" id="Rectangle-1-Copy-4" stroke="#E8C48E" stroke-width="2" fill="#FFF9EB"></path> + <text id="id:-'nodejs'-price:" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal" fill="#8A704D"> + <tspan x="171" y="240">id: 'nodejs'</tspan> + <tspan x="171" y="255">price: 10</tspan> + </text> + <text id="books" font-family="OpenSans-Regular, Open Sans" font-size="18" font-weight="normal" fill="#8A704D"> + <tspan x="160" y="31">books</tspan> + </text> + <text id="cursor" font-family="OpenSans-Regular, Open Sans" font-size="24" font-weight="normal" fill="#8A704D"> + <tspan x="1" y="79">cursor</tspan> + </text> + <path id="Line" d="M92.5290935,61.9295774 L167.596553,69.6069312 L167.393069,71.5965528 L92.3256094,63.919199 L91.7151573,69.8880639 L78.5,61.5 L93.1395456,55.9607125 L92.5290935,61.9295774 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <path id="Line-Copy" d="M91.9207297,79.2074366 L166.941663,74.9448836 L167.055116,76.9416631 L92.0341831,81.2042161 L92.3745433,87.1945546 L78,81 L91.5803696,73.2170981 L91.9207297,79.2074366 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <text id="primaryKey" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" fill="#8A704D"> + <tspan x="66" y="51">primaryKey</tspan> + </text> + <text id="key" font-family="OpenSans-Regular, Open Sans" font-size="16" font-weight="normal" fill="#8A704D"> + <tspan x="93" y="101">key</tspan> + </text> + </g> + </g> +</svg> \ No newline at end of file diff --git a/7-animation/1-bezier-curve/article.md b/7-animation/1-bezier-curve/article.md index 106a92dd5..63b9a03ce 100644 --- a/7-animation/1-bezier-curve/article.md +++ b/7-animation/1-bezier-curve/article.md @@ -2,7 +2,13 @@ Bezier curves are used in computer graphics to draw shapes, for CSS animation and in many other places. -They are actually a very simple thing, worth to study once and then feel comfortable in the world of vector graphics and advanced animations. +They are a very simple thing, worth to study once and then feel comfortable in the world of vector graphics and advanced animations. + +```smart header="Some theory, please" +This article provides a theoretical, but very needed insight into what Bezier curves are, while [the next one](info:css-animations#bezier-curve) shows how we can use them for CSS animations. + +Please take your time to read and understand the concept, it'll serve you well. +``` ## Control points @@ -31,7 +37,7 @@ For two points we have a linear curve (that's a straight line), for three points   -Because of that last property, in computer graphics it's possible to optimize intersection tests. If convex hulls do not intersect, then curves do not either. So checking for the convex hulls intersection first can give a very fast "no intersection" result. Checking the intersection or convex hulls is much easier, because they are rectangles, triangles and so on (see the picture above), much simpler figures than the curve. +Because of that last property, in computer graphics it's possible to optimize intersection tests. If convex hulls do not intersect, then curves do not either. So checking for the convex hulls intersection first can give a very fast "no intersection" result. Checking the intersection of convex hulls is much easier, because they are rectangles, triangles and so on (see the picture above), much simpler figures than the curve. **The main value of Bezier curves for drawing -- by moving the points the curve is changing *in intuitively obvious way*.** @@ -50,7 +56,7 @@ Here are some examples: ## De Casteljau's algorithm There's a mathematical formula for Bezier curves, but let's cover it a bit later, because -[De Casteljau's algorithm](https://en.wikipedia.org/wiki/De_Casteljau%27s_algorithm) it is identical to the mathematical definition and visually shows how it is constructed. +[De Casteljau's algorithm](https://en.wikipedia.org/wiki/De_Casteljau%27s_algorithm) is identical to the mathematical definition and visually shows how it is constructed. First let's see the 3-points example. @@ -130,14 +136,14 @@ A non-smooth Bezier curve (yeah, that's possible too): [iframe src="demo.svg?p=0,0,1,1,0,1,1,0&animate=1" height=370] ```online -If there's anything unclear in the algorithm description, then live examples above show how +If there's something unclear in the algorithm description, please look at the live examples above to see how the curve is built. ``` As the algorithm is recursive, we can build Bezier curves of any order, that is: using 5, 6 or more control points. But in practice many points are less useful. Usually we take 2-3 points, and for complex lines glue several curves together. That's simpler to develop and calculate. ```smart header="How to draw a curve *through* given points?" -We use control points for a Bezier curve. As we can see, they are not on the curve, except the first and the last ones. +To specify a Bezier curve, control points are used. As we can see, they are not on the curve, except the first and the last ones. Sometimes we have another task: to draw a curve *through several points*, so that all of them are on a single smooth curve. That task is called [interpolation](https://en.wikipedia.org/wiki/Interpolation), and here we don't cover it. @@ -176,7 +182,7 @@ Instead of <code>x<sub>1</sub>, y<sub>1</sub>, x<sub>2</sub>, y<sub>2</sub>, x<s For instance, if control points are `(0,0)`, `(0.5, 1)` and `(1, 0)`, the equations become: - <code>x = (1−t)<sup>2</sup> * 0 + 2(1−t)t * 0.5 + t<sup>2</sup> * 1 = (1-t)t + t<sup>2</sup> = t</code> -- <code>y = (1−t)<sup>2</sup> * 0 + 2(1−t)t * 1 + t<sup>2</sup> * 0 = 2(1-t)t = –t<sup>2</sup> + 2t</code> +- <code>y = (1−t)<sup>2</sup> * 0 + 2(1−t)t * 1 + t<sup>2</sup> * 0 = 2(1-t)t = –2t<sup>2</sup> + 2t</code> Now as `t` runs from `0` to `1`, the set of values `(x,y)` for each `t` forms the curve for such control points. @@ -186,12 +192,12 @@ Bezier curves are defined by their control points. We saw two definitions of Bezier curves: -1. Using a mathematical formulas. -2. Using a drawing process: De Casteljau's algorithm +1. Using a drawing process: De Casteljau's algorithm. +2. Using a mathematical formulas. Good properties of Bezier curves: -- We can draw smooth lines with a mouse by moving around control points. +- We can draw smooth lines with a mouse by moving control points. - Complex shapes can be made of several Bezier curves. Usage: diff --git a/7-animation/1-bezier-curve/bezier3-draw1.svg b/7-animation/1-bezier-curve/bezier3-draw1.svg index fd3ca092f..9fb69bbaf 100644 --- a/7-animation/1-bezier-curve/bezier3-draw1.svg +++ b/7-animation/1-bezier-curve/bezier3-draw1.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="340" height="350" viewBox="0 0 340 350"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-draw1.svg"><path id="Path-8" stroke="#DBAF88" d="M37.282 314.328L171.238 46.82l134.966 268.494"/><path id="Path-7" stroke="#A7333A" stroke-width="2" d="M37.738 314.328c98.312-197.562 188.187-157.862 268.157 1.49"/><circle id="Oval-1" cx="37" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="33" y="337">1</tspan></text><circle id="Oval-3" cx="306" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="302" y="337">3</tspan></text><circle id="Oval-2" cx="171" cy="47" r="4" fill="#FFF" stroke="#DBAF88"/><circle id="Oval-4" cx="104" cy="215" r="4" fill="#A7333A"/><text id="2" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="167" y="39">2</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="100" y="238">0.25</tspan></text><text id="t-=-0.25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="12" y="254.849">t = 0.25</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="209" y="112">0.25</tspan></text><path id="Line" stroke="#1C85B5" stroke-linecap="square" stroke-width="2" d="M70.5 247.5l134-133"/><image id="ease-in" width="133" height="135" x="231" y="-133" xlink:href=""/></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="340" height="350" viewBox="0 0 340 350"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-draw1.svg"><path id="Path-8" stroke="#DBAF88" d="M37.282 314.328L171.238 46.82l134.966 268.494"/><path id="Path-7" stroke="#A7333A" stroke-width="2" d="M37.738 314.328c98.312-197.562 188.187-157.862 268.157 1.49"/><circle id="Oval-1" cx="37" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="33" y="337">1</tspan></text><circle id="Oval-3" cx="306" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="302" y="337">3</tspan></text><circle id="Oval-2" cx="171" cy="47" r="4" fill="#FFF" stroke="#DBAF88"/><circle id="Oval-4" cx="104" cy="215" r="4" fill="#A7333A"/><text id="2" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="167" y="39">2</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="100" y="238">0.25</tspan></text><text id="t-=-0.25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="12" y="254.849">t = 0.25</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="209" y="112">0.25</tspan></text><path id="Line" stroke="#1C85B5" stroke-linecap="square" stroke-width="2" d="M70.5 247.5l134-133"/><image id="ease-in" width="133" height="135" x="231" y="-133" xlink:href=""/></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="340" height="350" viewBox="0 0 340 350"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-draw1.svg"><path id="Path-8" stroke="#DBAF88" d="M37.282 314.328L171.238 46.82l134.966 268.494"/><path id="Path-7" stroke="#A7333A" stroke-width="2" d="M37.738 314.328c98.312-197.562 188.187-157.862 268.157 1.49"/><circle id="Oval-1" cx="37" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="33" y="337">1</tspan></text><circle id="Oval-3" cx="306" cy="316" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="302" y="337">3</tspan></text><circle id="Oval-2" cx="171" cy="47" r="4" fill="#FFF" stroke="#DBAF88"/><circle id="Oval-4" cx="104" cy="215" r="4" fill="#A7333A"/><text id="2" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="167" y="39">2</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="100" y="238">0.25</tspan></text><text id="t-=-0.25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="12" y="254.849">t = 0.25</tspan></text><text id=".25" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="209" y="112">0.25</tspan></text><path id="Line" stroke="#1C85B5" stroke-linecap="square" stroke-width="2" d="M70.5 247.5l134-133"/><image id="ease-in" width="133" height="135" x="231" y="-133" xlink:href=""/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/7-animation/1-bezier-curve/bezier3-e.svg b/7-animation/1-bezier-curve/bezier3-e.svg index 4c5c741dc..35a72767a 100644 --- a/7-animation/1-bezier-curve/bezier3-e.svg +++ b/7-animation/1-bezier-curve/bezier3-e.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-e.svg"><path id="Triangle-1" stroke="#DBAF88" d="M74.5 43L125 144H24z"/><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="-4" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="70.19" y="36">3</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-e.svg"><path id="Triangle-1" stroke="#DBAF88" d="M74.5 43L125 144H24z"/><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="-4" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="70.19" y="36">3</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3-e.svg"><path id="Triangle-1" stroke="#DBAF88" d="M74.5 43L125 144H24z"/><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="2" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="70.19" y="36">2</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/7-animation/1-bezier-curve/bezier3.svg b/7-animation/1-bezier-curve/bezier3.svg index 35f1eb6e3..988655054 100644 --- a/7-animation/1-bezier-curve/bezier3.svg +++ b/7-animation/1-bezier-curve/bezier3.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3.svg"><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="69.99" y="32">2</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3.svg"><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="2" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="14" font-weight="normal"><tspan x="69.99" y="32">2</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="149" height="187" viewBox="0 0 149 187"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="animation" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="bezier3.svg"><path id="Path-4" stroke="#A7333A" stroke-width="2" d="M24.279 143.124c50.221-100.184 89.93-17.718 99.822.521"/><circle id="Oval-1" cx="24" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="1" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="20" y="165">1</tspan></text><circle id="Oval-2" cx="124" cy="144" r="4" fill="#FFF" stroke="#DBAF88"/><text id="3" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="120" y="165">3</tspan></text><circle id="Oval-3" cx="74" cy="44" r="4" fill="#FFF" stroke="#DBAF88"/><text id="2" fill="#AF6E24" font-family="OpenSans-Bold, Open Sans" font-size="14" font-weight="bold"><tspan x="70.19" y="36">2</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/7-animation/1-bezier-curve/demo.svg b/7-animation/1-bezier-curve/demo.svg index 5240697ee..50ff5e4c4 100644 --- a/7-animation/1-bezier-curve/demo.svg +++ b/7-animation/1-bezier-curve/demo.svg @@ -153,6 +153,12 @@ http://www.w3.org/Graphics/SVG/IG/resources/svgprimer.html#path_C points[i].y = y; setPointCoords(point, i); drawPath(); +<<<<<<< HEAD +======= + if (t > 0) { + drawT(points, t - STEP); + } +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b } document.onmouseup = function() { document.onmousemove = document.onmouseup = null; @@ -212,6 +218,10 @@ http://www.w3.org/Graphics/SVG/IG/resources/svgprimer.html#path_C } } +<<<<<<< HEAD +======= + const STEP = 0.005; +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b let t = 0; let timer; @@ -241,7 +251,11 @@ http://www.w3.org/Graphics/SVG/IG/resources/svgprimer.html#path_C return; } +<<<<<<< HEAD t += 0.005; +======= + t += STEP; +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b }, 30); } diff --git a/7-animation/2-css-animations/1-animate-logo-css/solution.view/index.html b/7-animation/2-css-animations/1-animate-logo-css/solution.view/index.html index 4e90e2478..d77f25e28 100644 --- a/7-animation/2-css-animations/1-animate-logo-css/solution.view/index.html +++ b/7-animation/2-css-animations/1-animate-logo-css/solution.view/index.html @@ -27,12 +27,12 @@ <img id="flyjet" src="https://en.js.cx/clipart/flyjet.jpg"> <script> - flyjet.onclick = function() { + let ended = false; // will change to true after the animation finishes - let ended = false; + flyjet.onclick = function() { flyjet.addEventListener('transitionend', function() { - if (!ended) { + if (!ended) { // check to show the message only once ended = true; alert('Done!'); } diff --git a/7-animation/2-css-animations/2-animate-logo-bezier-css/task.md b/7-animation/2-css-animations/2-animate-logo-bezier-css/task.md index 9e21f4101..18e63b480 100644 --- a/7-animation/2-css-animations/2-animate-logo-bezier-css/task.md +++ b/7-animation/2-css-animations/2-animate-logo-bezier-css/task.md @@ -4,7 +4,7 @@ importance: 5 # Animate the flying plane (CSS) -Modify the solution of the previous task <info:task/animate-logo-css> to make the plane grow more than it's original size 400x240px (jump out), and then return to that size. +Modify the solution of the previous task <info:task/animate-logo-css> to make the plane grow more than its original size 400x240px (jump out), and then return to that size. Here's how it should look (click on the plane): diff --git a/7-animation/2-css-animations/3-animate-circle/solution.view/index.html b/7-animation/2-css-animations/3-animate-circle/solution.view/index.html index 88b154254..43b927e8a 100644 --- a/7-animation/2-css-animations/3-animate-circle/solution.view/index.html +++ b/7-animation/2-css-animations/3-animate-circle/solution.view/index.html @@ -5,7 +5,7 @@ <meta charset="utf-8"> <style> .circle { - transition-property: width, height, margin-left, margin-top; + transition-property: width, height; transition-duration: 2s; position: fixed; transform: translateX(-50%) translateY(-50%); diff --git a/7-animation/2-css-animations/3-animate-circle/source.view/index.html b/7-animation/2-css-animations/3-animate-circle/source.view/index.html index a423280de..e305f4cf6 100644 --- a/7-animation/2-css-animations/3-animate-circle/source.view/index.html +++ b/7-animation/2-css-animations/3-animate-circle/source.view/index.html @@ -5,7 +5,7 @@ <meta charset="utf-8"> <style> .circle { - transition-property: width, height, margin-left, margin-top; + transition-property: width, height; transition-duration: 2s; position: fixed; transform: translateX(-50%) translateY(-50%); diff --git a/1-js/11-async/05-promise-api/02-promise-errors-as-results-2/solution.md b/7-animation/2-css-animations/4-animate-circle-callback/solution.md similarity index 100% rename from 1-js/11-async/05-promise-api/02-promise-errors-as-results-2/solution.md rename to 7-animation/2-css-animations/4-animate-circle-callback/solution.md diff --git a/1-js/11-async/01-callbacks/01-animate-circle-callback/solution.view/index.html b/7-animation/2-css-animations/4-animate-circle-callback/solution.view/index.html similarity index 94% rename from 1-js/11-async/01-callbacks/01-animate-circle-callback/solution.view/index.html rename to 7-animation/2-css-animations/4-animate-circle-callback/solution.view/index.html index b2192681c..64746e85f 100644 --- a/1-js/11-async/01-callbacks/01-animate-circle-callback/solution.view/index.html +++ b/7-animation/2-css-animations/4-animate-circle-callback/solution.view/index.html @@ -10,7 +10,7 @@ text-align: center; } .circle { - transition-property: width, height, margin-left, margin-top; + transition-property: width, height; transition-duration: 2s; position: fixed; transform: translateX(-50%) translateY(-50%); diff --git a/1-js/11-async/01-callbacks/01-animate-circle-callback/task.md b/7-animation/2-css-animations/4-animate-circle-callback/task.md similarity index 100% rename from 1-js/11-async/01-callbacks/01-animate-circle-callback/task.md rename to 7-animation/2-css-animations/4-animate-circle-callback/task.md diff --git a/7-animation/2-css-animations/article.md b/7-animation/2-css-animations/article.md index 844a27587..a6a41eaeb 100644 --- a/7-animation/2-css-animations/article.md +++ b/7-animation/2-css-animations/article.md @@ -1,14 +1,14 @@ # CSS-animations -CSS animations allow to do simple animations without JavaScript at all. +CSS animations make it possible to do simple animations without JavaScript at all. -JavaScript can be used to control CSS animation and make it even better with a little of code. +JavaScript can be used to control CSS animations and make them even better, with little code. ## CSS transitions [#css-transition] The idea of CSS transitions is simple. We describe a property and how its changes should be animated. When the property changes, the browser paints the animation. -That is: all we need is to change the property. And the fluent transition is made by the browser. +That is, all we need is to change the property, and the fluid transition will be done by the browser. For instance, the CSS below animates changes of `background-color` for 3 seconds: @@ -47,7 +47,7 @@ There are 4 properties to describe CSS transitions: - `transition-timing-function` - `transition-delay` -We'll cover them in a moment, for now let's note that the common `transition` property allows to declare them together in the order: `property duration timing-function delay`, and also animate multiple properties at once. +We'll cover them in a moment, for now let's note that the common `transition` property allows declaring them together in the order: `property duration timing-function delay`, as well as animating multiple properties at once. For instance, this button animates both `color` and `font-size`: @@ -70,25 +70,25 @@ growing.onclick = function() { </script> ``` -Now let's cover animation properties one by one. +Now, let's cover animation properties one by one. ## transition-property -In `transition-property` we write a list of property to animate, for instance: `left`, `margin-left`, `height`, `color`. +In `transition-property`, we write a list of properties to animate, for instance: `left`, `margin-left`, `height`, `color`. Or we could write `all`, which means "animate all properties". -Not all properties can be animated, but [many of them](http://www.w3.org/TR/css3-transitions/#animatable-properties-). The value `all` means "animate all properties". +Do note that, there are properties which can not be animated. However, [most of the generally used properties are animatable](https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_animated_properties). ## transition-duration -In `transition-duration` we can specify how long the animation should take. The time should be in [CSS time format](http://www.w3.org/TR/css3-values/#time): in seconds `s` or milliseconds `ms`. +In `transition-duration` we can specify how long the animation should take. The time should be in [CSS time format](https://www.w3.org/TR/css3-values/#time): in seconds `s` or milliseconds `ms`. ## transition-delay -In `transition-delay` we can specify the delay *before* the animation. For instance, if `transition-delay: 1s`, then animation starts after 1 second after the change. +In `transition-delay` we can specify the delay *before* the animation. For instance, if `transition-delay` is `1s` and `transition-duration` is `2s`, then the animation starts 1 second after the property change and the total duration will be 2 seconds. -Negative values are also possible. Then the animation starts from the middle. For instance, if `transition-duration` is `2s`, and the delay is `-1s`, then the animation takes 1 second and starts from the half. +Negative values are also possible. Then the animation is shown immediately, but the starting point of the animation will be after given value (time). For example, if `transition-delay` is `-1s` and `transition-duration` is `2s`, then animation starts from the halfway point and total duration will be 1 second. -Here's the animation shifts numbers from `0` to `9` using CSS `translate` property: +Here the animation shifts numbers from `0` to `9` using CSS `translate` property: [codetabs src="digits"] @@ -108,13 +108,13 @@ In the example above JavaScript adds the class `.animate` to the element -- and stripe.classList.add('animate'); ``` -We can also start it "from the middle", from the exact number, e.g. corresponding to the current second, using the negative `transition-delay`. +We could also start it from somewhere in the middle of the transition, from an exact number, e.g. corresponding to the current second, using a negative `transition-delay`. Here if you click the digit -- it starts the animation from the current second: [codetabs src="digits-negative-delay"] -JavaScript does it by an extra line: +JavaScript does it with an extra line: ```js stripe.onclick = function() { @@ -129,25 +129,25 @@ stripe.onclick = function() { ## transition-timing-function -Timing function describes how the animation process is distributed along the time. Will it start slowly and then go fast or vise versa. +The timing function describes how the animation process is distributed along its timeline. Will it start slowly and then go fast, or vice versa. -That's the most complicated property from the first sight. But it becomes very simple if we devote a bit time to it. +It appears to be the most complicated property at first. But it becomes very simple if we devote a bit time to it. -That property accepts two kinds of values: a Bezier curve or steps. Let's start from the curve, as it's used more often. +That property accepts two kinds of values: a Bezier curve or steps. Let's start with the curve, as it's used more often. ### Bezier curve -The timing function can be set as a [Bezier curve](/bezier-curve) with 4 control points that satisfies the conditions: +The timing function can be set as a [Bezier curve](/bezier-curve) with 4 control points that satisfy the conditions: 1. First control point: `(0,0)`. 2. Last control point: `(1,1)`. -3. For intermediate points values of `x` must be in the interval `0..1`, `y` can be anything. +3. For intermediate points, the values of `x` must be in the interval `0..1`, `y` can be anything. The syntax for a Bezier curve in CSS: `cubic-bezier(x2, y2, x3, y3)`. Here we need to specify only 2nd and 3rd control points, because the 1st one is fixed to `(0,0)` and the 4th one is `(1,1)`. -The timing function describes how fast the animation process goes in time. +The timing function describes how fast the animation process goes. -- The `x` axis is the time: `0` -- the starting moment, `1` -- the last moment of `transition-duration`. +- The `x` axis is the time: `0` -- the start, `1` -- the end of `transition-duration`. - The `y` axis specifies the completion of the process: `0` -- the starting value of the property, `1` -- the final value. The simplest variant is when the animation goes uniformly, with the same linear speed. That can be specified by the curve `cubic-bezier(0, 0, 1, 1)`. @@ -168,7 +168,7 @@ The CSS `transition` is based on that curve: .train { left: 0; transition: left 5s cubic-bezier(0, 0, 1, 1); - /* JavaScript sets left to 450px */ + /* click on a train sets left to 450px, thus triggering the animation */ } ``` @@ -191,13 +191,13 @@ CSS: .train { left: 0; transition: left 5s cubic-bezier(0, .5, .5, 1); - /* JavaScript sets left to 450px */ + /* click on a train sets left to 450px, thus triggering the animation */ } ``` There are several built-in curves: `linear`, `ease`, `ease-in`, `ease-out` and `ease-in-out`. -The `linear` is a shorthand for `cubic-bezier(0, 0, 1, 1)` -- a straight line, we saw it already. +The `linear` is a shorthand for `cubic-bezier(0, 0, 1, 1)` -- a straight line, which we described above. Other names are shorthands for the following `cubic-bezier`: @@ -210,27 +210,27 @@ Other names are shorthands for the following `cubic-bezier`: So we could use `ease-out` for our slowing down train: - ```css .train { left: 0; transition: left 5s ease-out; - /* transition: left 5s cubic-bezier(0, .5, .5, 1); */ + /* same as transition: left 5s cubic-bezier(0, .5, .5, 1); */ } ``` But it looks a bit differently. -**A Bezier curve can make the animation "jump out" of its range.** +**A Bezier curve can make the animation exceed its range.** -The control points on the curve can have any `y` coordinates: even negative or huge. Then the Bezier curve would also jump very low or high, making the animation go beyond its normal range. +The control points on the curve can have any `y` coordinates: even negative or huge ones. Then the Bezier curve would also extend very low or high, making the animation go beyond its normal range. In the example below the animation code is: + ```css .train { left: 100px; transition: left 5s cubic-bezier(.5, -1, .5, 2); - /* JavaScript sets left to 400px */ + /* click on a train sets left to 450px */ } ``` @@ -244,21 +244,29 @@ But if you click the train, you'll see that: [codetabs src="train-over"] -Why it happens -- pretty obvious if we look at the graph of the given Bezier curve: +Why it happens is pretty obvious if we look at the graph of the given Bezier curve:  -We moved the `y` coordinate of the 2nd point below zero, and for the 3rd point we made put it over `1`, so the curve goes out of the "regular" quadrant. The `y` is out of the "standard" range `0..1`. +We moved the `y` coordinate of the 2nd point below zero, and for the 3rd point we made it over `1`, so the curve goes out of the "regular" quadrant. The `y` is out of the "standard" range `0..1`. -As we know, `y` measures "the completion of the animation process". The value `y = 0` corresponds to the starting property value and `y = 1` -- the ending value. So values `y<0` move the property lower than the starting `left` and `y>1` -- over the final `left`. +As we know, `y` measures "the completion of the animation process". The value `y = 0` corresponds to the starting property value and `y = 1` -- the ending value. So values `y<0` move the property beyond the starting `left` and `y>1` -- past the final `left`. That's a "soft" variant for sure. If we put `y` values like `-99` and `99` then the train would jump out of the range much more. -But how to make the Bezier curve for a specific task? There are many tools. For instance, we can do it on the site <http://cubic-bezier.com/>. +But how do we make a Bezier curve for a specific task? There are many tools. + +- For instance, we can do it on the site <https://cubic-bezier.com>. +- Browser developer tools also have special support for Bezier curves in CSS: + 1. Open the developer tools with `key:F12` (Mac: `key:Cmd+Opt+I`). + 2. Select the `Elements` tab, then pay attention to the `Styles` sub-panel at the right side. + 3. CSS properties with a word `cubic-bezier` will have an icon before this word. + 4. Click this icon to edit the curve. + ### Steps -Timing function `steps(number of steps[, start/end])` allows to split animation into steps. +The timing function `steps(number of steps[, start/end])` allows splitting an transition into multiple steps. Let's see that in an example with digits. @@ -266,7 +274,19 @@ Here's a list of digits, without any animations, just as a source: [codetabs src="step-list"] -We'll make the digits appear in a discrete way by making the part of the list outside of the red "window" invisible and shifting the list to the left with each step. +In the HTML, a stripe of digits is enclosed into a fixed-length `<div id="digits">`: + +```html +<div id="digit"> + <div id="stripe">0123456789</div> +</div> +``` + +The `#digit` div has a fixed width and a border, so it looks like a red window. + +We'll make a timer: the digits will appear one by one, in a discrete way. + +To achieve that, we'll hide the `#stripe` outside of `#digit` using `overflow: hidden`, and then shift the `#stripe` to the left step-by-step. There will be 9 steps, a step-move for each digit: @@ -277,58 +297,60 @@ There will be 9 steps, a step-move for each digit: } ``` -In action: - -[codetabs src="step"] - The first argument of `steps(9, start)` is the number of steps. The transform will be split into 9 parts (10% each). The time interval is automatically divided into 9 parts as well, so `transition: 9s` gives us 9 seconds for the whole animation – 1 second per digit. The second argument is one of two words: `start` or `end`. -The `start` means that in the beginning of animation we need to do make the first step immediately. +The `start` means that in the beginning of animation we need to make the first step immediately. + +In action: + +[codetabs src="step"] -We can observe that during the animation: when we click on the digit it changes to `1` (the first step) immediately, and then changes in the beginning of the next second. +A click on the digit changes it to `1` (the first step) immediately, and then changes in the beginning of the next second. The process is progressing like this: - `0s` -- `-10%` (first change in the beginning of the 1st second, immediately) - `1s` -- `-20%` - ... -- `8s` -- `-80%` +- `8s` -- `-90%` - (the last second shows the final value). +Here, the first change was immediate because of `start` in the `steps`. + The alternative value `end` would mean that the change should be applied not in the beginning, but at the end of each second. -So the process would go like this: +So the process for `steps(9, end)` would go like this: -- `0s` -- `0` +- `0s` -- `0` (during the first second nothing changes) - `1s` -- `-10%` (first change at the end of the 1st second) - `2s` -- `-20%` - ... - `9s` -- `-90%` -Here's `step(9, end)` in action (note the pause between the first digit change): +Here's `steps(9, end)` in action (note the pause before the first digit change): [codetabs src="step-end"] -There are also shorthand values: +There are also some pre-defined shorthands for `steps(...)`: - `step-start` -- is the same as `steps(1, start)`. That is, the animation starts immediately and takes 1 step. So it starts and finishes immediately, as if there were no animation. - `step-end` -- the same as `steps(1, end)`: make the animation in a single step at the end of `transition-duration`. -These values are rarely used, because that's not really animation, but rather a single-step change. +These values are rarely used, as they represent not a real animation, but rather a single-step change. We mention them here for completeness. -## Event transitionend +## Event: "transitionend" -When the CSS animation finishes the `transitionend` event triggers. +When the CSS animation finishes, the `transitionend` event triggers. It is widely used to do an action after the animation is done. Also we can join animations. -For instance, the ship in the example below starts to swim there and back on click, each time farther and farther to the right: +For instance, the ship in the example below starts to sail there and back when clicked, each time farther and farther to the right: [iframe src="boat" height=300 edit link] -The animation is initiated by the function `go` that re-runs each time when the transition finishes and flips the direction: +The animation is initiated by the function `go` that re-runs each time the transition finishes, and flips the direction: ```js boat.onclick = function() { @@ -337,11 +359,11 @@ boat.onclick = function() { function go() { if (times % 2) { - // swim to the right + // sail to the right boat.classList.remove('back'); boat.style.marginLeft = 100 * times + 200 + 'px'; } else { - // swim to the left + // sail to the left boat.classList.add('back'); boat.style.marginLeft = 100 * times - 200 + 'px'; } @@ -357,7 +379,7 @@ boat.onclick = function() { }; ``` -The event object for `transitionend` has few specific properties: +The event object for `transitionend` has a few specific properties: `event.propertyName` : The property that has finished animating. Can be good if we animate multiple properties simultaneously. @@ -369,7 +391,7 @@ The event object for `transitionend` has few specific properties: We can join multiple simple animations together using the `@keyframes` CSS rule. -It specifies the "name" of the animation and rules: what, when and where to animate. Then using the `animation` property we attach the animation to the element and specify additional parameters for it. +It specifies the "name" of the animation and rules - what, when and where to animate. Then using the `animation` property, we can attach the animation to the element and specify additional parameters for it. Here's an example with explanations: @@ -405,11 +427,92 @@ Here's an example with explanations: There are many articles about `@keyframes` and a [detailed specification](https://drafts.csswg.org/css-animations/). -Probably you won't need `@keyframes` often, unless everything is in the constant move on your sites. +You probably won't need `@keyframes` often, unless everything is in constant motion on your sites. + +## Performance + +Most CSS properties can be animated, because most of them are numeric values. For instance, `width`, `color`, `font-size` are all numbers. When you animate them, the browser gradually changes these numbers frame by frame, creating a smooth effect. + +However, not all animations will look as smooth as you'd like, because different CSS properties cost differently to change. + +In more technical details, when there's a style change, the browser goes through 3 steps to render the new look: + +1. **Layout**: re-compute the geometry and position of each element, then +2. **Paint**: re-compute how everything should look like at their places, including background, colors, +3. **Composite**: render the final results into pixels on screen, apply CSS transforms if they exist. + +During a CSS animation, this process repeats every frame. However, CSS properties that never affect geometry or position, such as `color`, may skip the Layout step. If a `color` changes, the browser doesn't calculate any new geometry, it goes to Paint -> Composite. And there are few properties that directly go to Composite. You can find a longer list of CSS properties and which stages they trigger at <https://csstriggers.com>. + +The calculations may take time, especially on pages with many elements and a complex layout. And the delays are actually visible on most devices, leading to "jittery", less fluid animations. + +Animations of properties that skip the Layout step are faster. It's even better if Paint is skipped too. + +The `transform` property is a great choice, because: +- CSS transforms affect the target element box as a whole (rotate, flip, stretch, shift it). +- CSS transforms never affect neighbour elements. + +...So browsers apply `transform` "on top" of existing Layout and Paint calculations, in the Composite stage. + +In other words, the browser calculates the Layout (sizes, positions), paints it with colors, backgrounds, etc at the Paint stage, and then applies `transform` to element boxes that need it. + +Changes (animations) of the `transform` property never trigger Layout and Paint steps. More than that, the browser leverages the graphics accelerator (a special chip on the CPU or graphics card) for CSS transforms, thus making them very efficient. + +Luckily, the `transform` property is very powerful. By using `transform` on an element, you could rotate and flip it, stretch and shrink it, move it around, and [much more](https://developer.mozilla.org/docs/Web/CSS/transform#syntax). So instead of `left/margin-left` properties we can use `transform: translateX(…)`, use `transform: scale` for increasing element size, etc. + +The `opacity` property also never triggers Layout (also skips Paint in Mozilla Gecko). We can use it for show/hide or fade-in/fade-out effects. + +Paring `transform` with `opacity` can usually solve most of our needs, providing fluid, good-looking animations. + +For example, here clicking on the `#boat` element adds the class with `transform: translateX(300px)` and `opacity: 0`, thus making it move `300px` to the right and disappear: + +```html run height=260 autorun no-beautify +<img src="https://js.cx/clipart/boat.png" id="boat"> + +<style> +#boat { + cursor: pointer; + transition: transform 2s ease-in-out, opacity 2s ease-in-out; +} + +.move { + transform: translateX(300px); + opacity: 0; +} +</style> +<script> + boat.onclick = () => boat.classList.add('move'); +</script> +``` + +Here's a more complex example, with `@keyframes`: + +```html run height=80 autorun no-beautify +<h2 onclick="this.classList.toggle('animated')">click me to start / stop</h2> +<style> + .animated { + animation: hello-goodbye 1.8s infinite; + width: fit-content; + } + @keyframes hello-goodbye { + 0% { + transform: translateY(-60px) rotateX(0.7turn); + opacity: 0; + } + 50% { + transform: none; + opacity: 1; + } + 100% { + transform: translateX(230px) rotateZ(90deg) scale(0.5); + opacity: 0; + } + } +</style> +``` ## Summary -CSS animations allow to smoothly (or not) animate changes of one or multiple CSS properties. +CSS animations allow smoothly (or step-by-step) animated changes of one or multiple CSS properties. They are good for most animation tasks. We're also able to use JavaScript for animations, the next chapter is devoted to that. @@ -419,9 +522,11 @@ Limitations of CSS animations compared to JavaScript animations: + Simple things done simply. + Fast and lightweight for CPU. - JavaScript animations are flexible. They can implement any animation logic, like an "explosion" of an element. -- Not just property changes. We can create new elements in JavaScript for purposes of animation. +- Not just property changes. We can create new elements in JavaScript as part of the animation. ``` -The majority of animations can be implemented using CSS as described in this chapter. And `transitionend` event allows to run JavaScript after the animation, so it integrates fine with the code. +In early examples in this chapter, we animate `font-size`, `left`, `width`, `height`, etc. In real life projects, we should use `transform: scale()` and `transform: translate()` for better performance. + +The majority of animations can be implemented using CSS as described in this chapter. And the `transitionend` event allows JavaScript to be run after the animation, so it integrates fine with the code. But in the next chapter we'll do some JavaScript animations to cover more complex cases. diff --git a/7-animation/3-js-animation/1-animate-ball/solution.md b/7-animation/3-js-animation/1-animate-ball/solution.md index 3a9b8564f..0dc67b8bd 100644 --- a/7-animation/3-js-animation/1-animate-ball/solution.md +++ b/7-animation/3-js-animation/1-animate-ball/solution.md @@ -1,10 +1,8 @@ To bounce we can use CSS property `top` and `position:absolute` for the ball inside the field with `position:relative`. -The bottom coordinate of the field is `field.clientHeight`. But the `top` property gives coordinates for the top of the ball, the edge position is `field.clientHeight - ball.clientHeight`. +The bottom coordinate of the field is `field.clientHeight`. The CSS `top` property refers to the upper edge of the ball. So it should go from `0` till `field.clientHeight - ball.clientHeight`, that's the final lowest position of the upper edge of the ball. -So we animate the `top` from `0` to `field.clientHeight - ball.clientHeight`. - -Now to get the "bouncing" effect we can use the timing function `bounce` in `easeOut` mode. +To get the "bouncing" effect we can use the timing function `bounce` in `easeOut` mode. Here's the final code for the animation: diff --git a/7-animation/3-js-animation/1-animate-ball/solution.view/index.html b/7-animation/3-js-animation/1-animate-ball/solution.view/index.html index 7e031e8d1..146033cf7 100644 --- a/7-animation/3-js-animation/1-animate-ball/solution.view/index.html +++ b/7-animation/3-js-animation/1-animate-ball/solution.view/index.html @@ -21,7 +21,7 @@ } function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/7-animation/3-js-animation/2-animate-ball-hops/solution.view/index.html b/7-animation/3-js-animation/2-animate-ball-hops/solution.view/index.html index b246f422f..f587ff607 100644 --- a/7-animation/3-js-animation/2-animate-ball-hops/solution.view/index.html +++ b/7-animation/3-js-animation/2-animate-ball-hops/solution.view/index.html @@ -21,7 +21,7 @@ } function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/7-animation/3-js-animation/article.md b/7-animation/3-js-animation/article.md index b367ec09f..9a65912ff 100644 --- a/7-animation/3-js-animation/article.md +++ b/7-animation/3-js-animation/article.md @@ -55,7 +55,7 @@ Let's imagine we have several animations running simultaneously. If we run them separately, then even though each one has `setInterval(..., 20)`, then the browser would have to repaint much more often than every `20ms`. -That's because they have different starting time, so "every 20ms" differs between different animations. The intervals are not alignned. So we'll have several independent runs within `20ms`. +That's because they have different starting time, so "every 20ms" differs between different animations. The intervals are not aligned. So we'll have several independent runs within `20ms`. In other words, this: @@ -75,11 +75,11 @@ setInterval(animate2, 20); // in different places of the script setInterval(animate3, 20); ``` -These several independent redraws should be grouped together, to make the redraw easier for the browser (and hence smoother for people). +These several independent redraws should be grouped together, to make the redraw easier for the browser and hence load less CPU load and look smoother. -There's one more thing to keep in mind. Sometimes when CPU is overloaded, or there are other reasons to redraw less often (like when the browser tab is hidden), so we really shouldn't run it every `20ms`. +There's one more thing to keep in mind. Sometimes CPU is overloaded, or there are other reasons to redraw less often (like when the browser tab is hidden), so we really shouldn't run it every `20ms`. -But how do we know about that in JavaScript? There's a specification [Animation timing](http://www.w3.org/TR/animation-timing/) that provides the function `requestAnimationFrame`. It addresses all these issues and even more. +But how do we know about that in JavaScript? There's a specification [Animation timing](https://www.w3.org/TR/animation-timing/) that provides the function `requestAnimationFrame`. It addresses all these issues and even more. The syntax: ```js @@ -96,7 +96,7 @@ The returned value `requestId` can be used to cancel the call: cancelAnimationFrame(requestId); ``` -The `callback` gets one argument -- the time passed from the beginning of the page load in microseconds. This time can also be obtained by calling [performance.now()](mdn:api/Performance/now). +The `callback` gets one argument -- the time passed from the beginning of the page load in milliseconds. This time can also be obtained by calling [performance.now()](mdn:api/Performance/now). Usually `callback` runs very soon, unless the CPU is overloaded or the laptop battery is almost discharged, or there's another reason. @@ -159,7 +159,11 @@ Function `animate` accepts 3 parameters that essentially describes the animation } ``` +<<<<<<< HEAD It's graph: +======= + Its graph: +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b  That's just like `transition-timing-function: linear`. There are more interesting variants shown below. @@ -227,7 +231,7 @@ See in action (click to activate): [iframe height=40 src="quad" link] -...Or the cubic curve or event greater `n`. Increasing the power makes it speed up faster. +...Or the cubic curve or even greater `n`. Increasing the power makes it speed up faster. Here's the graph for `progress` in the power `5`: @@ -283,7 +287,7 @@ The `bounce` function does the same, but in the reverse order: "bouncing" starts ```js function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } @@ -397,7 +401,7 @@ The effect is clearly seen if we compare the graphs of `easeIn`, `easeOut` and `  -- <span style="color:#EE6B47">Red</span> is the regular variantof `circ` (`easeIn`). +- <span style="color:#EE6B47">Red</span> is the regular variant of `circ` (`easeIn`). - <span style="color:#8DB173">Green</span> -- `easeOut`. - <span style="color:#62C0DC">Blue</span> -- `easeInOut`. @@ -405,7 +409,7 @@ As we can see, the graph of the first half of the animation is the scaled down ` ## More interesting "draw" -Instead of moving the element we can do something else. All we need is to write the write the proper `draw`. +Instead of moving the element we can do something else. All we need is to write the proper `draw`. Here's the animated "bouncing" text typing: @@ -452,4 +456,4 @@ Surely we could improve it, add more bells and whistles, but JavaScript animatio JavaScript animations can use any timing function. We covered a lot of examples and transformations to make them even more versatile. Unlike CSS, we are not limited to Bezier curves here. -The same is about `draw`: we can animate anything, not just CSS properties. +The same is true about `draw`: we can animate anything, not just CSS properties. diff --git a/7-animation/3-js-animation/bounce-easeinout.view/index.html b/7-animation/3-js-animation/bounce-easeinout.view/index.html index 837c50db1..aed3d9d08 100644 --- a/7-animation/3-js-animation/bounce-easeinout.view/index.html +++ b/7-animation/3-js-animation/bounce-easeinout.view/index.html @@ -26,7 +26,7 @@ function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/7-animation/3-js-animation/bounce-easeout.view/index.html b/7-animation/3-js-animation/bounce-easeout.view/index.html index e52eae8de..69dbb7ce0 100644 --- a/7-animation/3-js-animation/bounce-easeout.view/index.html +++ b/7-animation/3-js-animation/bounce-easeout.view/index.html @@ -22,7 +22,7 @@ } function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/7-animation/3-js-animation/bounce.view/index.html b/7-animation/3-js-animation/bounce.view/index.html index 1be2580d9..3575ed820 100644 --- a/7-animation/3-js-animation/bounce.view/index.html +++ b/7-animation/3-js-animation/bounce.view/index.html @@ -19,7 +19,7 @@ animate({ duration: 3000, timing: function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/7-animation/3-js-animation/text.view/index.html b/7-animation/3-js-animation/text.view/index.html index e404fe5c4..4947e4cd4 100644 --- a/7-animation/3-js-animation/text.view/index.html +++ b/7-animation/3-js-animation/text.view/index.html @@ -29,14 +29,14 @@ timing: bounce, draw: function(progress) { let result = (to - from) * progress + from; - textArea.value = text.substr(0, Math.ceil(result)) + textArea.value = text.slice(0, Math.ceil(result)) } }); } function bounce(timeFraction) { - for (let a = 0, b = 1, result; 1; a += b, b /= 2) { + for (let a = 0, b = 1; 1; a += b, b /= 2) { if (timeFraction >= (7 - 4 * a) / 11) { return -Math.pow((11 - 6 * a - 11 * timeFraction) / 4, 2) + Math.pow(b, 2) } diff --git a/8-web-components/1-webcomponents-intro/article.md b/8-web-components/1-webcomponents-intro/article.md index f9427462b..c3522dea9 100644 --- a/8-web-components/1-webcomponents-intro/article.md +++ b/8-web-components/1-webcomponents-intro/article.md @@ -26,9 +26,9 @@ The International Space Station: ...And this thing flies, keeps humans alive in space! -How such complex devices are created? +How are such complex devices created? -Which principles we could borrow to make our development same-level reliable and scalable? Or, at least, close to it. +Which principles could we borrow to make our development same-level reliable and scalable? Or, at least, close to it? ## Component architecture @@ -57,14 +57,14 @@ Components may have subcomponents, e.g. messages may be parts of a higher-level How do we decide, what is a component? That comes from intuition, experience and common sense. Usually it's a separate visual entity that we can describe in terms of what it does and how it interacts with the page. In the case above, the page has blocks, each of them plays its own role, it's logical to make these components. A component has: -- its own JavaScript class. +- Its own JavaScript class. - DOM structure, managed solely by its class, outside code doesn't access it ("encapsulation" principle). - CSS styles, applied to the component. - API: events, class methods etc, to interact with other components. Once again, the whole "component" thing is nothing special. -There exist many frameworks and development methodologies to build them, each one with its own bells and whistles. Usually, special CSS classes and conventions are used to provide "component feel" -- CSS scoping and DOM encapsulation. +There exist many frameworks and development methodologies to build them, each with its own bells and whistles. Usually, special CSS classes and conventions are used to provide "component feel" -- CSS scoping and DOM encapsulation. "Web components" provide built-in browser capabilities for that, so we don't have to emulate them any more. diff --git a/8-web-components/1-webcomponents-intro/web-components-twitter.svg b/8-web-components/1-webcomponents-intro/web-components-twitter.svg index 9d3b0b00b..344e6420d 100644 --- a/8-web-components/1-webcomponents-intro/web-components-twitter.svg +++ b/8-web-components/1-webcomponents-intro/web-components-twitter.svg @@ -1 +1,5 @@ -<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="716" height="399" viewBox="0 0 716 399"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="web-components-twitter.svg"><image id="Снимок-экрана-2019-03-28-в-20.29.32" width="717" height="407" x="0" y="0" xlink:href=""/><path id="Rectangle" stroke="#C06334" stroke-width="3" d="M1.5 1.5h714v33H1.5z"/><path id="Rectangle-Copy" stroke="#C06334" stroke-width="3" d="M1.5 43.5h230v169H1.5z"/><path id="Rectangle-Copy-2" stroke="#C06334" stroke-width="3" d="M241.5 95.5h473v128h-473z"/><path id="Rectangle-Copy-3" stroke="#C06334" stroke-width="3" d="M241.5 333.5h473v64h-473z"/><path id="Rectangle-Copy-6" stroke="#C06334" stroke-width="3" d="M241.5 228.5h473v101h-473z"/><path id="Rectangle-Copy-4" stroke="#C06334" stroke-width="3" d="M714.5 47.5v43h-473v-43h473z"/><path id="Rectangle-Copy-5" stroke="#C06334" stroke-width="3" d="M1.5 221.5h230v176H1.5z"/><circle id="Oval-5-Copy" cx="18" cy="17" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="1" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="22">1</tspan></text><circle id="Oval-5-Copy" cx="18" cy="60" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="2" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="65">2</tspan></text><circle id="Oval-5-Copy-2" cx="259" cy="65" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="4" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="70">4</tspan></text><circle id="Oval-5-Copy-3" cx="19" cy="238" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="3-copy" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="14" y="243">3</tspan></text><circle id="Oval-5-Copy" cx="259" cy="113" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="5" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="118">5</tspan></text><circle id="Oval-5-Copy-4" cx="259" cy="246" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="6" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="251">6</tspan></text><circle id="Oval-5-Copy-5" cx="259" cy="351" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="7" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="356">7</tspan></text></g></g></svg> \ No newline at end of file +<<<<<<< HEAD +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="716" height="399" viewBox="0 0 716 399"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="web-components-twitter.svg"><image id="Снимок-экрана-2019-03-28-в-20.29.32" width="717" height="407" x="0" y="0" xlink:href=""/><path id="Rectangle" stroke="#C06334" stroke-width="3" d="M1.5 1.5h714v33H1.5z"/><path id="Rectangle-Copy" stroke="#C06334" stroke-width="3" d="M1.5 43.5h230v169H1.5z"/><path id="Rectangle-Copy-2" stroke="#C06334" stroke-width="3" d="M241.5 95.5h473v128h-473z"/><path id="Rectangle-Copy-3" stroke="#C06334" stroke-width="3" d="M241.5 333.5h473v64h-473z"/><path id="Rectangle-Copy-6" stroke="#C06334" stroke-width="3" d="M241.5 228.5h473v101h-473z"/><path id="Rectangle-Copy-4" stroke="#C06334" stroke-width="3" d="M714.5 47.5v43h-473v-43h473z"/><path id="Rectangle-Copy-5" stroke="#C06334" stroke-width="3" d="M1.5 221.5h230v176H1.5z"/><circle id="Oval-5-Copy" cx="18" cy="17" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="1" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="22">1</tspan></text><circle id="Oval-5-Copy" cx="18" cy="60" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="2" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="65">2</tspan></text><circle id="Oval-5-Copy-2" cx="259" cy="65" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="4" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="70">4</tspan></text><circle id="Oval-5-Copy-3" cx="19" cy="238" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="3-copy" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="14" y="243">3</tspan></text><circle id="Oval-5-Copy" cx="259" cy="113" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="5" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="118">5</tspan></text><circle id="Oval-5-Copy-4" cx="259" cy="246" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="6" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="251">6</tspan></text><circle id="Oval-5-Copy-5" cx="259" cy="351" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="7" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="356">7</tspan></text></g></g></svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="716" height="399" viewBox="0 0 716 399"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="dom" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="web-components-twitter.svg"><image id="Снимок-экрана-2019-03-28-в-20.29.32" width="717" height="407" x="0" y="0" xlink:href=""/><path id="Rectangle" stroke="#C06334" stroke-width="3" d="M1.5 1.5h714v33H1.5z"/><path id="Rectangle-Copy" stroke="#C06334" stroke-width="3" d="M1.5 43.5h230v169H1.5z"/><path id="Rectangle-Copy-2" stroke="#C06334" stroke-width="3" d="M241.5 95.5h473v128h-473z"/><path id="Rectangle-Copy-3" stroke="#C06334" stroke-width="3" d="M241.5 333.5h473v64h-473z"/><path id="Rectangle-Copy-6" stroke="#C06334" stroke-width="3" d="M241.5 228.5h473v101h-473z"/><path id="Rectangle-Copy-4" stroke="#C06334" stroke-width="3" d="M714.5 47.5v43h-473v-43h473z"/><path id="Rectangle-Copy-5" stroke="#C06334" stroke-width="3" d="M1.5 221.5h230v176H1.5z"/><circle id="Oval-5-Copy" cx="18" cy="17" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="1" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="22">1</tspan></text><circle id="Oval-5-Copy" cx="18" cy="60" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="2" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="13" y="65">2</tspan></text><circle id="Oval-5-Copy-2" cx="259" cy="65" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="4" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="70">4</tspan></text><circle id="Oval-5-Copy-3" cx="19" cy="238" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="3-copy" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="14" y="243">3</tspan></text><circle id="Oval-5-Copy" cx="259" cy="113" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="5" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="118">5</tspan></text><circle id="Oval-5-Copy-4" cx="259" cy="246" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="6" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="251">6</tspan></text><circle id="Oval-5-Copy-5" cx="259" cy="351" r="12" fill="#FBF2EC" stroke="#DBAF88" stroke-width="2"/><text id="7" fill="#181717" font-family="PTMono-Bold, PT Mono" font-size="18" font-weight="bold"><tspan x="254" y="356">7</tspan></text></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/8-web-components/2-custom-elements/article.md b/8-web-components/2-custom-elements/article.md index 85434c75a..a84ed1192 100644 --- a/8-web-components/2-custom-elements/article.md +++ b/8-web-components/2-custom-elements/article.md @@ -3,7 +3,7 @@ We can create custom HTML elements, described by our class, with its own methods and properties, events and so on. -Once an custom element is defined, we can use it on par with built-in HTML elements. +Once a custom element is defined, we can use it on par with built-in HTML elements. That's great, as HTML dictionary is rich, but not infinite. There are no `<easy-tabs>`, `<sliding-carousel>`, `<beautiful-upload>`... Just think of any other tag we might need. @@ -12,9 +12,9 @@ We can define them with a special class, and then use as if they were always a p There are two kinds of custom elements: 1. **Autonomous custom elements** -- "all-new" elements, extending the abstract `HTMLElement` class. -2. **Customized built-in elements** -- extending built-in elements, like customized `HTMLButtonElement` etc. +2. **Customized built-in elements** -- extending built-in elements, like a customized button, based on `HTMLButtonElement` etc. -First we'll create autonomous elements, and then customized built-in ones. +First we'll cover autonomous elements, and then move to customized built-in ones. To create a custom element, we need to tell the browser several details about it: how to show it, what to do when the element is added or removed to page, etc. @@ -30,12 +30,12 @@ class MyElement extends HTMLElement { } connectedCallback() { - // browser calls it when the element is added to the document + // browser calls this method when the element is added to the document // (can be called many times if an element is repeatedly added/removed) } disconnectedCallback() { - // browser calls it when the element is removed from the document + // browser calls this method when the element is removed from the document // (can be called many times if an element is repeatedly added/removed) } @@ -115,7 +115,7 @@ customElements.define("time-formatted", TimeFormatted); // (2) ></time-formatted> ``` -1. The class has only one method `connectedCallback()` -- the browser calls it when `<time-formatted>` element is added to page (or when HTML parser detects it), and it uses the built-in [Intl.DateTimeFormat](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat) data formatter, well-supported across the browsers, to show a nicely formatted time. +1. The class has only one method `connectedCallback()` -- the browser calls it when `<time-formatted>` element is added to page (or when HTML parser detects it), and it uses the built-in [Intl.DateTimeFormat](mdn:/JavaScript/Reference/Global_Objects/DateTimeFormat) data formatter, well-supported across the browsers, to show a nicely formatted time. 2. We need to register our new element by `customElements.define(tag, class)`. 3. And then we can use it everywhere. @@ -138,7 +138,7 @@ In the example above, element content is rendered (created) in `connectedCallbac Why not in the `constructor`? -The reason is simple: when `constructor` is called, it's yet too early. The element instance is created, but not populated yet. The browser did not yet process/assign attributes at this stage: calls to `getAttribute` would return `null`. So we can't really render there. +The reason is simple: when `constructor` is called, it's yet too early. The element is created, but the browser did not yet process/assign attributes at this stage: calls to `getAttribute` would return `null`. So we can't really render there. Besides, if you think about it, that's better performance-wise -- to delay the work until it's really needed. @@ -149,7 +149,7 @@ The `connectedCallback` triggers when the element is added to the document. Not In the current implementation of `<time-formatted>`, after the element is rendered, further attribute changes don't have any effect. That's strange for an HTML element. Usually, when we change an attribute, like `a.href`, we expect the change to be immediately visible. So let's fix this. -We can observe attributes by providing their list in `observedAttributes()` static getter. For such attributes, `attributeChangedCallback` is called when they are modified. It doesn't trigger for an attribute for performance reasons. +We can observe attributes by providing their list in `observedAttributes()` static getter. For such attributes, `attributeChangedCallback` is called when they are modified. It doesn't trigger for other, unlisted attributes (that's for performance reasons). Here's a new `<time-formatted>`, that auto-updates when attributes change: @@ -242,7 +242,7 @@ customElements.define('user-info', class extends HTMLElement { If you run it, the `alert` is empty. -That's exactly because there are no children on that stage, the DOM is unfinished. HTML parser connected the custom element `<user-info>`, and will now proceed to its children, but just didn't yet. +That's exactly because there are no children on that stage, the DOM is unfinished. HTML parser connected the custom element `<user-info>`, and is going to proceed to its children, but just didn't yet. If we'd like to pass information to custom element, we can use attributes. They are available immediately. @@ -297,12 +297,12 @@ Output order: 1. outer connected. 2. inner connected. -2. outer initialized. +3. outer initialized. 4. inner initialized. -We can clearly see that the outer element does not wait for the inner one. +We can clearly see that the outer element finishes initialization `(3)` before the inner one `(4)`. -There's no built-in callback that triggers after nested elements are ready. But we can implement such thing on our own. For instance, inner elements can dispatch events like `initialized`, and outer ones can listen and react on them. +There's no built-in callback that triggers after nested elements are ready. If needed, we can implement such thing on our own. For instance, inner elements can dispatch events like `initialized`, and outer ones can listen and react on them. ## Customized built-in elements @@ -310,7 +310,7 @@ New elements that we create, such as `<time-formatted>`, don't have any associat But such things can be important. E.g, a search engine would be interested to know that we actually show a time. And if we're making a special kind of button, why not reuse the existing `<button>` functionality? -We can extend and customize built-in elements by inheriting from their classes. +We can extend and customize built-in HTML elements by inheriting from their classes. For example, buttons are instances of `HTMLButtonElement`, let's build upon it. @@ -320,11 +320,12 @@ For example, buttons are instances of `HTMLButtonElement`, let's build upon it. class HelloButton extends HTMLButtonElement { /* custom element methods */ } ``` -2. Provide an third argument to `customElements.define`, that specifies the tag: +2. Provide the third argument to `customElements.define`, that specifies the tag: ```js customElements.define('hello-button', HelloButton, *!*{extends: 'button'}*/!*); ``` - There exist different tags that share the same class, that's why it's needed. + + There may be different tags that share the same DOM-class, that's why specifying `extends` is needed. 3. At the end, to use our custom element, insert a regular `<button>` tag, but add `is="hello-button"` to it: ```html @@ -364,7 +365,7 @@ Our new button extends the built-in one. So it keeps the same styles and standar ## References - HTML Living Standard: <https://html.spec.whatwg.org/#custom-elements>. -- Compatiblity: <https://caniuse.com/#feat=custom-elements>. +- Compatiblity: <https://caniuse.com/#feat=custom-elementsv1>. ## Summary @@ -396,4 +397,4 @@ Custom elements can be of two types: /* <button is="my-button"> */ ``` -Custom elements are well-supported among browsers. Edge is a bit behind, but there's a polyfill <https://github.com/webcomponents/webcomponentsjs>. +Custom elements are well-supported among browsers. There's a polyfill <https://github.com/webcomponents/polyfills/tree/master/packages/webcomponentsjs>. diff --git a/8-web-components/3-shadow-dom/article.md b/8-web-components/3-shadow-dom/article.md index 7a58dddb3..92614f777 100644 --- a/8-web-components/3-shadow-dom/article.md +++ b/8-web-components/3-shadow-dom/article.md @@ -35,9 +35,9 @@ input::-webkit-slider-runnable-track { <input type="range"> ``` -Once again, `pseudo` is a non-standard attribute. Chronologically, browsers first started to experiment with internal DOM structures to implement controls, and then, after time, shadow DOM was standartized to allow us, developers, to do the similar thing. +Once again, `pseudo` is a non-standard attribute. Chronologically, browsers first started to experiment with internal DOM structures to implement controls, and then, after time, shadow DOM was standardized to allow us, developers, to do the similar thing. -Furhter on, we'll use the modern shadow DOM standard, covered by [DOM spec](https://dom.spec.whatwg.org/#shadow-trees) other related specifications. +Further on, we'll use the modern shadow DOM standard, covered by [DOM spec](https://dom.spec.whatwg.org/#shadow-trees) and other related specifications. ## Shadow tree @@ -98,7 +98,7 @@ alert(elem.shadowRoot.host === elem); // true Shadow DOM is strongly delimited from the main document: -1. Shadow DOM elements are not visible to `querySelector` from the light DOM. In particular, Shadow DOM elements may have ids that conflict with those in the light DOM. They be unique only within the shadow tree. +1. Shadow DOM elements are not visible to `querySelector` from the light DOM. In particular, Shadow DOM elements may have ids that conflict with those in the light DOM. They must be unique only within the shadow tree. 2. Shadow DOM has own stylesheets. Style rules from the outer DOM don't get applied. For example: diff --git a/8-web-components/4-template-element/article.md b/8-web-components/4-template-element/article.md index 4d9df9cb3..5499c4edc 100644 --- a/8-web-components/4-template-element/article.md +++ b/8-web-components/4-template-element/article.md @@ -1,9 +1,9 @@ # Template element -A built-in `<template>` element serves as a storage for markup. The browser ignores it contents, only checks for syntax validity, but we can access and use it in JavaScript, to create other elements. +A built-in `<template>` element serves as a storage for HTML markup templates. The browser ignores its contents, only checks for syntax validity, but we can access and use it in JavaScript, to create other elements. -In theory, we could create any invisible element somewhere in HTML for markup storage purposes. What's special about `<template>`? +In theory, we could create any invisible element somewhere in HTML for HTML markup storage purposes. What's special about `<template>`? First, its content can be any valid HTML, even if it normally requires a proper enclosing tag. @@ -31,9 +31,9 @@ We can put styles and scripts into `<template>` as well: </template> ``` -The browser considers `<template>` content "out of the document", so the style is not applied, scripts are executed, `<video autoplay>` is not run, etc. +The browser considers `<template>` content "out of the document": styles are not applied, scripts are not executed, `<video autoplay>` is not run, etc. -The content becomes live (the script executes) when we insert it. +The content becomes live (styles apply, scripts run etc) when we insert it into the document. ## Inserting template @@ -87,7 +87,7 @@ Let's rewrite a Shadow DOM example from the previous chapter using `<template>`: </script> ``` -In the line `(*)` when we clone and insert `tmpl.content`, its children (`<style>`, `<p>`) are inserted instead. +In the line `(*)` when we clone and insert `tmpl.content`, as its `DocumentFragment`, its children (`<style>`, `<p>`) are inserted instead. They form the shadow DOM: @@ -109,8 +109,8 @@ To summarize: The `<template>` tag is quite unique, because: -- The browser checks the syntax inside it (as opposed to using a template string inside a script). -- ...But still allows to use any top-level HTML tags, even those that don't make sense without proper wrappers (e.g. `<tr>`). +- The browser checks HTML syntax inside it (as opposed to using a template string inside a script). +- ...But still allows use of any top-level HTML tags, even those that don't make sense without proper wrappers (e.g. `<tr>`). - The content becomes interactive: scripts run, `<video autoplay>` plays etc, when inserted into the document. -The `<template>` tag does not feature any sophisticated iteration mechanisms, data binding or variable substitutions, making it less powerful than frameworks. But we can build those on top of it. +The `<template>` element does not feature any iteration mechanisms, data binding or variable substitutions, but we can implement those on top of it. diff --git a/8-web-components/5-slots-composition/article.md b/8-web-components/5-slots-composition/article.md index d90fbc98f..c41e26e05 100644 --- a/8-web-components/5-slots-composition/article.md +++ b/8-web-components/5-slots-composition/article.md @@ -58,7 +58,7 @@ customElements.define('user-card', class extends HTMLElement { In the shadow DOM, `<slot name="X">` defines an "insertion point", a place where elements with `slot="X"` are rendered. -Then the browser performs "composition": it takes elements from the light DOM and renders them in corresponding slots of the shadow DOM. At the end, we have exactly what we want -- a generic component that can be filled with data. +Then the browser performs "composition": it takes elements from the light DOM and renders them in corresponding slots of the shadow DOM. At the end, we have exactly what we want -- a component that can be filled with data. Here's the DOM structure after the script, not taking composition into account: @@ -76,7 +76,7 @@ Here's the DOM structure after the script, not taking composition into account: </user-card> ``` -There's nothing odd here. We created the shadow DOM, so here it is. Now the element has both light and shadow DOM. +We created the shadow DOM, so here it is, under `#shadow-root`. Now the element has both light and shadow DOM. For rendering purposes, for each `<slot name="...">` in shadow DOM, the browser looks for `slot="..."` with the same name in the light DOM. These elements are rendered inside the slots: @@ -89,7 +89,7 @@ The result is called "flattened" DOM: #shadow-root <div>Name: <slot name="username"> - <!-- slotted element is inserted into the slot as a whole --> + <!-- slotted element is inserted into the slot --> <span slot="username">John Smith</span> </slot> </div> @@ -101,16 +101,16 @@ The result is called "flattened" DOM: </user-card> ``` -...But the "flattened" DOM is only created for rendering and event-handling purposes. That's how things are shown. The nodes are actually not moved around! +...But the flattened DOM exists only for rendering and event-handling purposes. It's kind of "virtual". That's how things are shown. But the nodes in the document are actually not moved around! -That can be easily checked if we run `querySelector`: nodes are still at their places. +That can be easily checked if we run `querySelectorAll`: nodes are still at their places. ```js // light DOM <span> nodes are still at the same place, under `<user-card>` -alert( document.querySelector('user-card span').length ); // 2 +alert( document.querySelectorAll('user-card span').length ); // 2 ``` -It may look bizarre, but for shadow DOM with slots we have one more "DOM level", the "flattened" DOM -- result of slot insertion. The browser renders it and uses for style inheritance, event propagation. But JavaScript still sees the document "as is", before flattening. +So, the flattened DOM is derived from shadow DOM by inserting slots. The browser renders it and uses for style inheritance, event propagation (more about that later). But JavaScript still sees the document "as is", before flattening. ````warn header="Only top-level children may have slot=\"...\" attribute" The `slot="..."` attribute is only valid for direct children of the shadow host (in our example, `<user-card>` element). For nested elements it's ignored. @@ -120,18 +120,44 @@ For example, the second `<span>` here is ignored (as it's not a top-level child <user-card> <span slot="username">John Smith</span> <div> - <!-- bad slot, not top-level: --> + <!-- invalid slot, must be direct child of user-card --> <span slot="birthday">01.01.2001</span> </div> </user-card> ``` - -In practice, there's no sense in slotting a deeply nested element, so this limitation just ensures the correct DOM structure. ```` +If there are multiple elements in light DOM with the same slot name, they are appended into the slot, one after another. + +For example, this: + +```html +<user-card> + <span slot="username">John</span> + <span slot="username">Smith</span> +</user-card> +``` + +Gives this flattened DOM with two elements in `<slot name="username">`: + +```html +<user-card> + #shadow-root + <div>Name: + <slot name="username"> + <span slot="username">John</span> + <span slot="username">Smith</span> + </slot> + </div> + <div>Birthday: + <slot name="birthday"></slot> + </div> +</user-card> +``` + ## Slot fallback content -If we put something inside a `<slot>`, it becomes the fallback content. The browser shows it if there's no corresponding filler in light DOM. +If we put something inside a `<slot>`, it becomes the fallback, "default" content. The browser shows it if there's no corresponding filler in light DOM. For example, in this piece of shadow DOM, `Anonymous` renders if there's no `slot="username"` in light DOM. @@ -141,11 +167,11 @@ For example, in this piece of shadow DOM, `Anonymous` renders if there's no `slo </div> ``` -## Default slot +## Default slot: first unnamed The first `<slot>` in shadow DOM that doesn't have a name is a "default" slot. It gets all nodes from the light DOM that aren't slotted elsewhere. -For example, let's add the default slot to our `<user-card>` that collects any unslotted information about the user: +For example, let's add the default slot to our `<user-card>` that shows all unslotted information about the user: ```html run autorun="no-epub" untrusted height=140 <script> @@ -202,11 +228,11 @@ The flattened DOM looks like this: </slot> </div> <fieldset> - <legend>About me</legend> + <legend>Other information</legend> *!* <slot> - <div>Hello</div> - <div>I am John!</div> + <div>I like to swim.</div> + <div>...And play volleyball too!</div> </slot> */!* </fieldset> @@ -243,7 +269,7 @@ The shadow DOM template with proper slots: ``` 1. `<span slot="title">` goes into `<slot name="title">`. -2. There are many `<li slot="item">` in the template, but only one `<slot name="item">` in the template. That's perfectly normal. All elements with `slot="item"` get appended to `<slot name="item">` one after another, thus forming the list. +2. There are many `<li slot="item">` in the `<custom-menu>`, but only one `<slot name="item">` in the template. So all such `<li slot="item">` are appended to `<slot name="item">` one after another, thus forming the list. The flattened DOM becomes: @@ -293,7 +319,7 @@ Here's the full demo: Of course, we can add more functionality to it: events, methods and so on. -## Monitoring slots +## Updating slots What if the outer code wants to add/remove menu items dynamically? @@ -301,7 +327,7 @@ What if the outer code wants to add/remove menu items dynamically? Also, as light DOM nodes are not copied, but just rendered in slots, the changes inside them immediately become visible. -So we don't have to do anything to update rendering. But if the component wants to know about slot changes, then `slotchange` event is available. +So we don't have to do anything to update rendering. But if the component code wants to know about slot changes, then `slotchange` event is available. For example, here the menu item is inserted dynamically after 1 second, and the title changes after 2 seconds: @@ -355,7 +381,7 @@ If we'd like to track internal modifications of light DOM from JavaScript, that' Finally, let's mention the slot-related JavaScript methods. -As we've seen before, JavaScript looks at the "real" DOM, without flattening. But, if the shadow tree has `{mode: 'open'}`, then we can figure out which elements assigned to a slot and, vise-versa, the slot by the element inside it: +As we've seen before, JavaScript looks at the "real" DOM, without flattening. But, if the shadow tree has `{mode: 'open'}`, then we can figure out which elements assigned to a slot and, vice-versa, the slot by the element inside it: - `node.assignedSlot` -- returns the `<slot>` element that the `node` is assigned to. - `slot.assignedNodes({flatten: true/false})` -- DOM nodes, assigned to the slot. The `flatten` option is `false` by default. If explicitly set to `true`, then it looks more deeply into the flattened DOM, returning nested slots in case of nested components and the fallback content if no node assigned. @@ -383,7 +409,7 @@ customElements.define('custom-menu', class extends HTMLElement { <ul><slot name="item"></slot></ul> </div>`; - // slottable is added/removed/replaced + // triggers when slot content changes *!* this.shadowRoot.firstElementChild.addEventListener('slotchange', e => { let slot = e.target; @@ -406,7 +432,7 @@ setTimeout(() => { ## Summary -Slots allow to show light DOM children in shadow DOM. +Usually, if an element has shadow DOM, then its light DOM is not displayed. Slots allow to show elements from light DOM in specified places of shadow DOM. There are two kinds of slots: @@ -421,12 +447,12 @@ Composition does not really move nodes, from JavaScript point of view the DOM is JavaScript can access slots using methods: - `slot.assignedNodes/Elements()` -- returns nodes/elements inside the `slot`. -- `node.assignedSlot` -- the reverse meethod, returns slot by a node. +- `node.assignedSlot` -- the reverse property, returns slot by a node. If we'd like to know what we're showing, we can track slot contents using: - `slotchange` event -- triggers the first time a slot is filled, and on any add/remove/replace operation of the slotted element, but not its children. The slot is `event.target`. - [MutationObserver](info:mutation-observer) to go deeper into slot content, watch changes inside it. -Now, as we have elements from light DOM in the shadow DOM, let's see how to style them properly. The basic rule is that shadow elements are styled inside, and light elements -- outside, but there are notable exceptions. +Now, as we know how to show elements from light DOM in shadow DOM, let's see how to style them properly. The basic rule is that shadow elements are styled inside, and light elements -- outside, but there are notable exceptions. We'll see the details in the next chapter. diff --git a/8-web-components/6-shadow-dom-style/article.md b/8-web-components/6-shadow-dom-style/article.md index 66e5cc13d..98e246a7f 100644 --- a/8-web-components/6-shadow-dom-style/article.md +++ b/8-web-components/6-shadow-dom-style/article.md @@ -1,6 +1,6 @@ # Shadow DOM styling -Shadow DOM may include both `<style>` and `<link rel="stylesheet" href="…">` tags. In the latter case, stylesheets are HTTP-cached, so they are not redownloaded. There's no overhead in @importing or linking same styles for many components. +Shadow DOM may include both `<style>` and `<link rel="stylesheet" href="…">` tags. In the latter case, stylesheets are HTTP-cached, so they are not redownloaded for multiple components that use same template. As a general rule, local styles work only inside the shadow tree, and document styles work outside of it. But there are few exceptions. @@ -44,7 +44,7 @@ customElements.define('custom-dialog', class extends HTMLElement { ## Cascading -The shadow host (`<custom-dialog>` itself) resides in the light DOM, so it's affected by the main CSS cascade. +The shadow host (`<custom-dialog>` itself) resides in the light DOM, so it's affected by document CSS rules. If there's a property styled both in `:host` locally, and in the document, then the document style takes precedence. @@ -58,7 +58,7 @@ custom-dialog { ``` ...Then the `<custom-dialog>` would be without padding. -It's very convenient, as we can setup "default" styles in the component `:host` rule, and then easily override them in the document. +It's very convenient, as we can setup "default" component styles in its `:host` rule, and then easily override them in the document. The exception is when a local property is labelled `!important`, for such properties, local styles take precedence. @@ -79,6 +79,7 @@ For example, we'd like to center the `<custom-dialog>` only if it has `centered` left: 50%; top: 50%; transform: translate(-50%, -50%); + border-color: blue; } :host { @@ -108,24 +109,9 @@ customElements.define('custom-dialog', class extends HTMLElement { </custom-dialog> ``` -Now the additional centering styles are only applied to the first dialog `<custom-dialog centered>`. +Now the additional centering styles are only applied to the first dialog: `<custom-dialog centered>`. -## :host-context(selector) - -Same as `:host`, but applied only if the shadow host or any of its ancestors in the outer document matches the `selector`. - -E.g. `:host-context(.dark-theme)` matches only if there's `dark-theme` class on `<custom-dialog>` on above it: - -```html -<body class="dark-theme"> - <!-- - :host-context(.dark-theme) applies to custom-dialogs inside .dark-theme - --> - <custom-dialog>...</custom-dialog> -</body> -``` - -To summarize, we can use `:host`-family of selectors to style the main element of the component, depending on the context. These styles (unless `!important`) can be overridden by the document. +To summarize, we can use `:host`-family of selectors to style the main element of the component. These styles (unless `!important`) can be overridden by the document. ## Styling slotted content @@ -190,11 +176,11 @@ customElements.define('user-card', class extends HTMLElement { </script> ``` -Here `<p>John Smith</p>` becomes bold, because CSS inheritance is in effect between the `<slot>` and its contents. But not all CSS properties are inherited. +Here `<p>John Smith</p>` becomes bold, because CSS inheritance is in effect between the `<slot>` and its contents. But in CSS itself not all properties are inherited. Another option is to use `::slotted(selector)` pseudo-class. It matches elements based on two conditions: -1. The element from the light DOM that is inserted into a `<slot>`. Then slot name doesn't matter. Just any slotted element, but only the element itself, not its children. +1. That's a slotted element, that comes from the light DOM. Slot name doesn't matter. Just any slotted element, but only the element itself, not its children. 2. The element matches the `selector`. In our example, `::slotted(div)` selects exactly `<div slot="username">`, but not its children: @@ -239,26 +225,25 @@ Also, `::slotted` can only be used in CSS. We can't use it in `querySelector`. ## CSS hooks with custom properties -How do we style a component in-depth from the main document? +How do we style internal elements of a component from the main document? -Naturally, document styles apply to `<custom-dialog>` element or `<user-card>`, etc. But how can we affect its internals? For instance, in `<user-card>` we'd like to allow the outer document change how user fields look. +Selectors like `:host` apply rules to `<custom-dialog>` element or `<user-card>`, but how to style shadow DOM elements inside them? -Just as we expose methods to interact with our component, we can expose CSS variables (custom CSS properties) to style it. +There's no selector that can directly affect shadow DOM styles from the document. But just as we expose methods to interact with our component, we can expose CSS variables (custom CSS properties) to style it. **Custom CSS properties exist on all levels, both in light and shadow.** -For example, in shadow DOM we can use `--user-card-field-color` CSS variable to style fields: +For example, in shadow DOM we can use `--user-card-field-color` CSS variable to style fields, and the outer document can set its value: ```html <style> .field { color: var(--user-card-field-color, black); - /* if --user-card-field-color is not defined, use black */ + /* if --user-card-field-color is not defined, use black color */ } </style> <div class="field">Name: <slot name="username"></slot></div> <div class="field">Birthday: <slot name="birthday"></slot></div> -</style> ``` Then, we can declare this property in the outer document for `<user-card>`: @@ -317,12 +302,12 @@ Shadow DOM can include styles, such as `<style>` or `<link rel="stylesheet">`. Local styles can affect: - shadow tree, -- shadow host with `:host`-family pseudoclasses, +- shadow host with `:host` and `:host()` pseudoclasses, - slotted elements (coming from light DOM), `::slotted(selector)` allows to select slotted elements themselves, but not their children. Document styles can affect: -- shadow host (as it's in the outer document) -- slotted elements and their contents (as it's physically in the outer document) +- shadow host (as it lives in the outer document) +- slotted elements and their contents (as that's also in the outer document) When CSS properties conflict, normally document styles have precedence, unless the property is labelled as `!important`. Then local styles have precedence. diff --git a/8-web-components/7-shadow-dom-events/article.md b/8-web-components/7-shadow-dom-events/article.md index 14e310a55..d1f627e3e 100644 --- a/8-web-components/7-shadow-dom-events/article.md +++ b/8-web-components/7-shadow-dom-events/article.md @@ -2,9 +2,9 @@ The idea behind shadow tree is to encapsulate internal implementation details of a component. -Let's say, a click event happens inside a shadow DOM of `<user-card>` component. But scripts in the main document have no idea about the shadow DOM internals, especially if the component comes from a 3rd-party library or so. +Let's say, a click event happens inside a shadow DOM of `<user-card>` component. But scripts in the main document have no idea about the shadow DOM internals, especially if the component comes from a 3rd-party library. -So, to keep things simple, the browser *retargets* the event. +So, to keep the details encapsulated, the browser *retargets* the event. **Events that happen in shadow DOM have the host element as the target, when caught outside of the component.** @@ -35,11 +35,11 @@ If you click on the button, the messages are: 1. Inner target: `BUTTON` -- internal event handler gets the correct target, the element inside shadow DOM. 2. Outer target: `USER-CARD` -- document event handler gets shadow host as the target. -Event retargeting is a great thing to have, because the outer document doesn't have no know about component internals. From its point of view, the event happened on `<user-card>`. +Event retargeting is a great thing to have, because the outer document doesn't have to know about component internals. From its point of view, the event happened on `<user-card>`. **Retargeting does not occur if the event occurs on a slotted element, that physically lives in the light DOM.** -For example, if a user clicks on `<span slot="username">` in the example below, the event target is exactly this element, for both shadow and light handlers: +For example, if a user clicks on `<span slot="username">` in the example below, the event target is exactly this `span` element, for both shadow and light handlers: ```html run autorun="no-epub" untrusted height=60 <user-card id="userCard"> @@ -75,7 +75,7 @@ For purposes of event bubbling, flattened DOM is used. So, if we have a slotted element, and an event occurs somewhere inside it, then it bubbles up to the `<slot>` and upwards. -The full path to the original event target, with all the shadow root elements, can be obtained using `event.composedPath()`. As we can see from the name of the method, that path is taken after the composition. +The full path to the original event target, with all the shadow elements, can be obtained using `event.composedPath()`. As we can see from the name of the method, that path is taken after the composition. In the example above, the flattened DOM is: @@ -119,7 +119,7 @@ All touch events and pointer events also have `composed: true`. There are some events that have `composed: false` though: -- `mouseenter`, `mouseleave` (they also do not bubble), +- `mouseenter`, `mouseleave` (they do not bubble at all), - `load`, `unload`, `abort`, `error`, - `select`, - `slotchange`. @@ -189,4 +189,4 @@ These events can be caught only on elements within the same DOM. If we dispatch a `CustomEvent`, then we should explicitly set `composed: true`. -Please note that in case of nested components, composed events bubble through all shadow DOM boundaries. So, if an event is intended only for the immediate enclosing component, we can also dispatch it on the shadow host. Then it's out of the shadow DOM already. +Please note that in case of nested components, one shadow DOM may be nested into another. In that case composed events bubble through all shadow DOM boundaries. So, if an event is intended only for the immediate enclosing component, we can also dispatch it on the shadow host and set `composed: false`. Then it's out of the component shadow DOM, but won't bubble up to higher-level DOM. diff --git a/9-regular-expressions/01-regexp-introduction/article.md b/9-regular-expressions/01-regexp-introduction/article.md index 59775d3c0..ade62cdf2 100644 --- a/9-regular-expressions/01-regexp-introduction/article.md +++ b/9-regular-expressions/01-regexp-introduction/article.md @@ -1,59 +1,65 @@ # Patterns and flags -Regular expressions is a powerful way of searching and replacing inside a string. +Regular expressions are patterns that provide a powerful way to search and replace in text. -In JavaScript regular expressions are implemented using objects of a built-in `RegExp` class and integrated with strings. +In JavaScript, they are available via the [RegExp](mdn:js/RegExp) object, as well as being integrated in methods of strings. -Please note that regular expressions vary between programming languages. In this tutorial we concentrate on JavaScript. Of course there's a lot in common, but they are a somewhat different in Perl, Ruby, PHP etc. - -## Regular expressions +## Regular Expressions A regular expression (also "regexp", or just "reg") consists of a *pattern* and optional *flags*. -There are two syntaxes to create a regular expression object. +There are two syntaxes that can be used to create a regular expression object. -The long syntax: +The "long" syntax: ```js regexp = new RegExp("pattern", "flags"); ``` -...And the short one, using slashes `"/"`: +And the "short" one, using slashes `"/"`: ```js regexp = /pattern/; // no flags regexp = /pattern/gmi; // with flags g,m and i (to be covered soon) ``` -Slashes `"/"` tell JavaScript that we are creating a regular expression. They play the same role as quotes for strings. +Slashes `pattern:/.../` tell JavaScript that we are creating a regular expression. They play the same role as quotes for strings. -## Usage +In both cases `regexp` becomes an instance of the built-in `RegExp` class. -To search inside a string, we can use method [search](mdn:js/String/search). +The main difference between these two syntaxes is that pattern using slashes `/.../` does not allow for expressions to be inserted (like string template literals with `${...}`). They are fully static. -Here's an example: +Slashes are used when we know the regular expression at the code writing time -- and that's the most common situation. While `new RegExp` is more often used when we need to create a regexp "on the fly" from a dynamically generated string. For instance: -```js run -let str = "I love JavaScript!"; // will search here +```js +let tag = prompt("What tag do you want to find?", "h2"); -let regexp = /love/; -alert( str.search(regexp) ); // 2 +let regexp = new RegExp(`<${tag}>`); // same as /<h2>/ if answered "h2" in the prompt above ``` -The `str.search` method looks for the pattern `pattern:/love/` and returns the position inside the string. As we might guess, `pattern:/love/` is the simplest possible pattern. What it does is a simple substring search. +## Flags -The code above is the same as: +Regular expressions may have flags that affect the search. -```js run -let str = "I love JavaScript!"; // will search here +There are only 6 of them in JavaScript: -let substr = 'love'; -alert( str.search(substr) ); // 2 -``` +`pattern:i` +: With this flag the search is case-insensitive: no difference between `A` and `a` (see the example below). -So searching for `pattern:/love/` is the same as searching for `"love"`. +`pattern:g` +: With this flag the search looks for all matches, without it -- only the first match is returned. -But that's only for now. Soon we'll create more complex regular expressions with much more searching power. +`pattern:m` +: Multiline mode (covered in the chapter <info:regexp-multiline-mode>). + +`pattern:s` +: Enables "dotall" mode, that allows a dot `pattern:.` to match newline character `\n` (covered in the chapter <info:regexp-character-classes>). + +`pattern:u` +: Enables full Unicode support. The flag enables correct processing of surrogate pairs. More about that in the chapter <info:regexp-unicode>. + +`pattern:y` +: "Sticky" mode: searching at the exact position in the text (covered in the chapter <info:regexp-sticky>) ```smart header="Colors" From here on the color scheme is: @@ -63,65 +69,109 @@ From here on the color scheme is: - result -- `match:green` ``` +## Searching: str.match -````smart header="When to use `new RegExp`?" -Normally we use the short syntax `/.../`. But it does not support variable insertions `${...}`. +As mentioned previously, regular expressions are integrated with string methods. -On the other hand, `new RegExp` allows to construct a pattern dynamically from a string, so it's more flexible. +The method `str.match(regexp)` finds all matches of `regexp` in the string `str`. -Here's an example of a dynamically generated regexp: +It has 3 working modes: -```js run -let tag = prompt("Which tag you want to search?", "h2"); -let regexp = new RegExp(`<${tag}>`); +1. If the regular expression has flag `pattern:g`, it returns an array of all matches: + ```js run + let str = "We will, we will rock you"; -// finds <h2> by default -alert( "<h1> <h2> <h3>".search(regexp)); -``` -```` + alert( str.match(/we/gi) ); // We,we (an array of 2 substrings that match) + ``` + Please note that both `match:We` and `match:we` are found, because flag `pattern:i` makes the regular expression case-insensitive. +2. If there's no such flag it returns only the first match in the form of an array, with the full match at index `0` and some additional details in properties: + ```js run + let str = "We will, we will rock you"; -## Flags + let result = str.match(/we/i); // without flag g -Regular expressions may have flags that affect the search. + alert( result[0] ); // We (1st match) + alert( result.length ); // 1 -There are only 5 of them in JavaScript: + // Details: + alert( result.index ); // 0 (position of the match) + alert( result.input ); // We will, we will rock you (source string) + ``` + The array may have other indexes, besides `0` if a part of the regular expression is enclosed in parentheses. We'll cover that in the chapter <info:regexp-groups>. -`i` -: With this flag the search is case-insensitive: no difference between `A` and `a` (see the example below). +3. And, finally, if there are no matches, `null` is returned (doesn't matter if there's flag `pattern:g` or not). -`g` -: With this flag the search looks for all matches, without it -- only the first one (we'll see uses in the next chapter). + This a very important nuance. If there are no matches, we don't receive an empty array, but instead receive `null`. Forgetting about that may lead to errors, e.g.: -`m` -: Multiline mode (covered in the chapter <info:regexp-multiline-mode>). + ```js run + let matches = "JavaScript".match(/HTML/); // = null + + if (!matches.length) { // Error: Cannot read property 'length' of null + alert("Error in the line above"); + } + ``` + + If we'd like the result to always be an array, we can write it this way: -`s` -: "Dotall" mode, allows `.` to match newlines (covered in the chapter <info:regexp-character-classes>). + ```js run + let matches = "JavaScript".match(/HTML/)*!* || []*/!*; -`u` -: Enables full unicode support. The flag enables correct processing of surrogate pairs. More about that in the chapter <info:regexp-unicode>. + if (!matches.length) { + alert("No matches"); // now it works + } + ``` -`y` -: Sticky mode (covered in the chapter <info:regexp-sticky>) +## Replacing: str.replace -We'll cover all these flags further in the tutorial. +The method `str.replace(regexp, replacement)` replaces matches found using `regexp` in string `str` with `replacement` (all matches if there's flag `pattern:g`, otherwise, only the first one). -For now, the simplest flag is `i`, here's an example: +For instance: ```js run -let str = "I love JavaScript!"; +// no flag g +alert( "We will, we will".replace(/we/i, "I") ); // I will, we will + +// with flag g +alert( "We will, we will".replace(/we/ig, "I") ); // I will, I will +``` + +The second argument is the `replacement` string. We can use special character combinations in it to insert fragments of the match: + +| Symbols | Action in the replacement string | +|--------|--------| +|`$&`|inserts the whole match| +|<code>$`</code>|inserts a part of the string before the match| +|`$'`|inserts a part of the string after the match| +|`$n`|if `n` is a 1-2 digit number, then it inserts the contents of n-th parentheses, more about it in the chapter <info:regexp-groups>| +|`$<name>`|inserts the contents of the parentheses with the given `name`, more about it in the chapter <info:regexp-groups>| +|`$$`|inserts character `$` | -alert( str.search(/LOVE/i) ); // 2 (found lowercased) +An example with `pattern:$&`: + +```js run +alert( "I love HTML".replace(/HTML/, "$& and JavaScript") ); // I love HTML and JavaScript +``` + +## Testing: regexp.test + +The method `regexp.test(str)` looks for at least one match, if found, returns `true`, otherwise `false`. + +```js run +let str = "I love JavaScript"; +let regexp = /LOVE/i; -alert( str.search(/LOVE/) ); // -1 (nothing found without 'i' flag) +alert( regexp.test(str) ); // true ``` -So the `i` flag already makes regular expressions more powerful than a simple substring search. But there's so much more. We'll cover other flags and features in the next chapters. +Later in this chapter we'll study more regular expressions, walk through more examples, and also meet other methods. +Full information about the methods is given in the article <info:regexp-methods>. ## Summary -- A regular expression consists of a pattern and optional flags: `g`, `i`, `m`, `u`, `s`, `y`. -- Without flags and special symbols that we'll study later, the search by a regexp is the same as a substring search. -- The method `str.search(regexp)` returns the index where the match is found or `-1` if there's no match. In the next chapter we'll see other methods. +- A regular expression consists of a pattern and optional flags: `pattern:g`, `pattern:i`, `pattern:m`, `pattern:u`, `pattern:s`, `pattern:y`. +- Without flags and special symbols (that we'll study later), the search by a regexp is the same as a substring search. +- The method `str.match(regexp)` looks for matches: all of them if there's `pattern:g` flag, otherwise, only the first one. +- The method `str.replace(regexp, replacement)` replaces matches found using `regexp` with `replacement`: all of them if there's `pattern:g` flag, otherwise only the first one. +- The method `regexp.test(str)` returns `true` if there's at least one match, otherwise, it returns `false`. diff --git a/9-regular-expressions/02-regexp-character-classes/article.md b/9-regular-expressions/02-regexp-character-classes/article.md new file mode 100644 index 000000000..201c78a05 --- /dev/null +++ b/9-regular-expressions/02-regexp-character-classes/article.md @@ -0,0 +1,203 @@ +# Character classes + +Consider a practical task -- we have a phone number like `"+7(903)-123-45-67"`, and we need to turn it into pure numbers: `79031234567`. + +To do so, we can find and remove anything that's not a number. Character classes can help with that. + +A *character class* is a special notation that matches any symbol from a certain set. + +For the start, let's explore the "digit" class. It's written as `pattern:\d` and corresponds to "any single digit". + +For instance, let's find the first digit in the phone number: + +```js run +let str = "+7(903)-123-45-67"; + +let regexp = /\d/; + +alert( str.match(regexp) ); // 7 +``` + +Without the flag `pattern:g`, the regular expression only looks for the first match, that is the first digit `pattern:\d`. + +Let's add the `pattern:g` flag to find all digits: + +```js run +let str = "+7(903)-123-45-67"; + +let regexp = /\d/g; + +alert( str.match(regexp) ); // array of matches: 7,9,0,3,1,2,3,4,5,6,7 + +// let's make the digits-only phone number of them: +alert( str.match(regexp).join('') ); // 79031234567 +``` + +That was a character class for digits. There are other character classes as well. + +Most used are: + +`pattern:\d` ("d" is from "digit") +: A digit: a character from `0` to `9`. + +`pattern:\s` ("s" is from "space") +: A space symbol: includes spaces, tabs `\t`, newlines `\n` and few other rare characters, such as `\v`, `\f` and `\r`. + +`pattern:\w` ("w" is from "word") +: A "wordly" character: either a letter of Latin alphabet or a digit or an underscore `_`. Non-Latin letters (like cyrillic or hindi) do not belong to `pattern:\w`. + +For instance, `pattern:\d\s\w` means a "digit" followed by a "space character" followed by a "wordly character", such as `match:1 a`. + +**A regexp may contain both regular symbols and character classes.** + +For instance, `pattern:CSS\d` matches a string `match:CSS` with a digit after it: + +```js run +let str = "Is there CSS4?"; +let regexp = /CSS\d/ + +alert( str.match(regexp) ); // CSS4 +``` + +Also we can use many character classes: + +```js run +alert( "I love HTML5!".match(/\s\w\w\w\w\d/) ); // ' HTML5' +``` + +The match (each regexp character class has the corresponding result character): + + + +## Inverse classes + +For every character class there exists an "inverse class", denoted with the same letter, but uppercased. + +The "inverse" means that it matches all other characters, for instance: + +`pattern:\D` +: Non-digit: any character except `pattern:\d`, for instance a letter. + +`pattern:\S` +: Non-space: any character except `pattern:\s`, for instance a letter. + +`pattern:\W` +: Non-wordly character: anything but `pattern:\w`, e.g a non-latin letter or a space. + +In the beginning of the chapter we saw how to make a number-only phone number from a string like `subject:+7(903)-123-45-67`: find all digits and join them. + +```js run +let str = "+7(903)-123-45-67"; + +alert( str.match(/\d/g).join('') ); // 79031234567 +``` + +An alternative, shorter way is to find non-digits `pattern:\D` and remove them from the string: + +```js run +let str = "+7(903)-123-45-67"; + +alert( str.replace(/\D/g, "") ); // 79031234567 +``` + +## A dot is "any character" + +A dot `pattern:.` is a special character class that matches "any character except a newline". + +For instance: + +```js run +alert( "Z".match(/./) ); // Z +``` + +Or in the middle of a regexp: + +```js run +let regexp = /CS.4/; + +alert( "CSS4".match(regexp) ); // CSS4 +alert( "CS-4".match(regexp) ); // CS-4 +alert( "CS 4".match(regexp) ); // CS 4 (space is also a character) +``` + +Please note that a dot means "any character", but not the "absence of a character". There must be a character to match it: + +```js run +alert( "CS4".match(/CS.4/) ); // null, no match because there's no character for the dot +``` + +### Dot as literally any character with "s" flag + +By default, a dot doesn't match the newline character `\n`. + +For instance, the regexp `pattern:A.B` matches `match:A`, and then `match:B` with any character between them, except a newline `\n`: + +```js run +alert( "A\nB".match(/A.B/) ); // null (no match) +``` + +There are many situations when we'd like a dot to mean literally "any character", newline included. + +That's what flag `pattern:s` does. If a regexp has it, then a dot `pattern:.` matches literally any character: + +```js run +alert( "A\nB".match(/A.B/s) ); // A\nB (match!) +``` + +````warn header="Not supported in IE" +The `pattern:s` flag is not supported in IE. + +Luckily, there's an alternative, that works everywhere. We can use a regexp like `pattern:[\s\S]` to match "any character" (this pattern will be covered in the article <info:regexp-character-sets-and-ranges>). + +```js run +alert( "A\nB".match(/A[\s\S]B/) ); // A\nB (match!) +``` + +The pattern `pattern:[\s\S]` literally says: "a space character OR not a space character". In other words, "anything". We could use another pair of complementary classes, such as `pattern:[\d\D]`, that doesn't matter. Or even the `pattern:[^]` -- as it means match any character except nothing. + +Also we can use this trick if we want both kind of "dots" in the same pattern: the actual dot `pattern:.` behaving the regular way ("not including a newline"), and also a way to match "any character" with `pattern:[\s\S]` or alike. +```` + +````warn header="Pay attention to spaces" +Usually we pay little attention to spaces. For us strings `subject:1-5` and `subject:1 - 5` are nearly identical. + +But if a regexp doesn't take spaces into account, it may fail to work. + +Let's try to find digits separated by a hyphen: + +```js run +alert( "1 - 5".match(/\d-\d/) ); // null, no match! +``` + +Let's fix it adding spaces into the regexp `pattern:\d - \d`: + +```js run +alert( "1 - 5".match(/\d - \d/) ); // 1 - 5, now it works +// or we can use \s class: +alert( "1 - 5".match(/\d\s-\s\d/) ); // 1 - 5, also works +``` + +**A space is a character. Equal in importance with any other character.** + +We can't add or remove spaces from a regular expression and expect it to work the same. + +In other words, in a regular expression all characters matter, spaces too. +```` + +## Summary + +There exist following character classes: + +- `pattern:\d` -- digits. +- `pattern:\D` -- non-digits. +- `pattern:\s` -- space symbols, tabs, newlines. +- `pattern:\S` -- all but `pattern:\s`. +- `pattern:\w` -- Latin letters, digits, underscore `'_'`. +- `pattern:\W` -- all but `pattern:\w`. +- `pattern:.` -- any character if with the regexp `'s'` flag, otherwise any except a newline `\n`. + +...But that's not all! + +Unicode encoding, used by JavaScript for strings, provides many properties for characters, like: which language the letter belongs to (if it's a letter), is it a punctuation sign, etc. + +We can search by these properties as well. That requires flag `pattern:u`, covered in the next article. diff --git a/9-regular-expressions/02-regexp-character-classes/love-html5-classes.svg b/9-regular-expressions/02-regexp-character-classes/love-html5-classes.svg new file mode 100644 index 000000000..60d31da35 --- /dev/null +++ b/9-regular-expressions/02-regexp-character-classes/love-html5-classes.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="236" height="38" viewBox="0 0 236 38"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="love-html5-classes.svg" fill-rule="nonzero"><path id="IloveHTML5" fill="#AF6E24" d="M11 19h10.538v1.66h-4.369v12.388h4.369v1.66H11v-1.66h4.347V20.66H11V19zm36.788 0h3.644v11.938c0 .913.15 1.556.45 1.93.3.374.743.561 1.328.561.41 0 .802-.075 1.175-.224.373-.15.787-.404 1.24-.763l.79 1.234c-.233.21-.49.396-.768.561a5.651 5.651 0 01-1.734.673c-.293.06-.564.09-.812.09a4.554 4.554 0 01-1.45-.213 2.47 2.47 0 01-1.064-.685c-.286-.314-.501-.729-.648-1.245-.146-.516-.22-1.156-.22-1.919V20.504h-1.931V19zm16.385 10.098c0-1.81.46-3.242 1.383-4.297.922-1.055 2.24-1.582 3.952-1.582.922 0 1.716.153 2.382.46a4.683 4.683 0 011.657 1.245c.44.524.765 1.145.977 1.863a8.12 8.12 0 01.319 2.311c0 .898-.118 1.71-.352 2.435-.234.726-.578 1.343-1.032 1.851a4.558 4.558 0 01-1.68 1.178c-.665.277-1.422.416-2.271.416-.908 0-1.698-.154-2.371-.46a4.66 4.66 0 01-1.669-1.246 5.115 5.115 0 01-.977-1.862 8.12 8.12 0 01-.318-2.312zm1.822 0c0 .524.062 1.047.187 1.571.124.524.325.995.603 1.414.278.419.64.755 1.087 1.01.446.254.992.381 1.636.381 1.17 0 2.052-.37 2.645-1.11.593-.741.89-1.83.89-3.266 0-.538-.063-1.066-.187-1.582a4.226 4.226 0 00-.615-1.402 3.266 3.266 0 00-1.098-1.01c-.446-.255-.991-.382-1.635-.382-1.171 0-2.05.367-2.635 1.1-.585.733-.878 1.825-.878 3.276zm21.303 3.434h.241l3.447-9.044h1.91l-4.632 11.22h-1.932l-4.72-11.22h2.02l3.666 9.044zm22.73.763c-.264.239-.575.463-.933.673-.359.21-.75.389-1.175.538a8.46 8.46 0 01-1.328.348 7.92 7.92 0 01-1.394.124c-.878 0-1.654-.139-2.327-.416a4.48 4.48 0 01-1.69-1.189 5.213 5.213 0 01-1.033-1.851c-.234-.718-.351-1.526-.351-2.424 0-.942.128-1.78.384-2.513.256-.733.626-1.347 1.109-1.84a4.816 4.816 0 011.756-1.133c.688-.262 1.456-.393 2.305-.393a6.86 6.86 0 011.823.247c.6.164 1.13.475 1.591.93.461.457.816 1.086 1.065 1.886.249.8.337 1.836.263 3.108h-8.496c0 1.346.355 2.352 1.065 3.018.71.666 1.658.999 2.843.999.395 0 .787-.049 1.175-.146a9.656 9.656 0 001.108-.348c.352-.135.663-.284.933-.449.271-.164.48-.314.626-.449l.68 1.28zm-4.633-8.595c-.483 0-.94.052-1.372.157-.432.105-.812.28-1.141.527-.33.247-.6.569-.813.965-.212.397-.347.887-.406 1.47h6.828c-.073-.987-.384-1.754-.933-2.3-.55-.546-1.27-.82-2.163-.82zm38.544 2.805h-6.893v7.203h-1.822V19h1.822v6.844h6.893V19h1.823v15.708h-1.823v-7.203zM152.531 19h11.416v4.107h-1.625V20.66h-3.183v12.387h2.415v1.66h-6.652v-1.66h2.415V20.66h-3.162v2.446h-1.624V19zm26.9 5.273l.22-2.131h-.11l-1.053 1.93-2.24 3.837h-.548l-2.35-3.86-1.01-1.907h-.11l.264 2.11v10.456h-1.778V19h1.69l3.645 5.97h.044l3.512-5.97h1.647v15.708h-1.822V24.273zm9.69-5.273h1.822v14.048h6.213v-3.905h1.625v5.565h-9.66V19zm21.39 14.407c.557 0 1.069-.071 1.537-.213a3.16 3.16 0 001.197-.651c.33-.292.585-.655.768-1.089.183-.433.275-.942.275-1.526 0-1.077-.377-1.896-1.131-2.457-.754-.56-1.782-.841-3.085-.841l-2.063.112V19h7.53v1.616h-5.884v4.555l1.098-.045c1.625 0 2.905.404 3.842 1.212.937.808 1.405 1.96 1.405 3.456 0 .838-.143 1.578-.428 2.221a4.677 4.677 0 01-1.175 1.627 4.976 4.976 0 01-1.767.999 7.079 7.079 0 01-2.228.337c-.44 0-.857-.034-1.252-.101a8.91 8.91 0 01-1.075-.247 6.429 6.429 0 01-.846-.314c-.241-.113-.413-.214-.516-.303l.703-1.459c.249.15.63.33 1.142.539.512.209 1.163.314 1.953.314z"/><path id="\s\w\w\w\w\d" fill="#C06334" d="M122.288 13.776l-1.056.464-5.92-13.184 1.072-.448 5.904 13.168zm4.563-3.936c0-.341-.133-.605-.4-.792a3.282 3.282 0 00-.992-.448 18.35 18.35 0 00-1.288-.312 7.43 7.43 0 01-1.288-.384 2.886 2.886 0 01-.992-.648c-.267-.272-.4-.653-.4-1.144 0-.405.088-.752.264-1.04.176-.288.408-.525.696-.712a3.181 3.181 0 011.008-.416c.384-.09.784-.136 1.2-.136.747 0 1.39.093 1.928.28.539.187.968.381 1.288.584l-.512 1.008a13.475 13.475 0 00-1.144-.552c-.41-.176-.925-.264-1.544-.264-.235 0-.467.024-.696.072a2.46 2.46 0 00-.624.216 1.244 1.244 0 00-.448.376.943.943 0 00-.168.568c0 .277.133.49.4.64.267.15.597.275.992.376.395.101.824.197 1.288.288.464.09.893.221 1.288.392.395.17.725.4.992.688.267.288.4.677.4 1.168 0 .736-.29 1.344-.872 1.824-.581.48-1.453.72-2.616.72-.352 0-.699-.032-1.04-.096a6.308 6.308 0 01-.96-.256 5.776 5.776 0 01-.816-.36 3.431 3.431 0 01-.608-.408l.64-1.04c.128.128.299.259.512.392a4.833 4.833 0 001.528.616 4.296 4.296 0 001.696.024c.25-.048.472-.123.664-.224.192-.101.344-.235.456-.4.112-.165.168-.365.168-.6zm12.966 3.936l-1.056.464-5.92-13.184 1.072-.448 5.904 13.168zm2.978-8.016l1.6 4.448h.112L145.643 4h1.168l-1.568 8h-1.424l-1.536-4.416h-.048l-1.6 4.416h-1.424l-1.68-8h1.232l1.296 6.24h.096l1.552-4.48h1.088zm14.55 8.016l-1.056.464-5.92-13.184 1.072-.448 5.904 13.168zm2.979-8.016l1.6 4.448h.112L163.172 4h1.168l-1.568 8h-1.424l-1.536-4.416h-.048l-1.6 4.416h-1.424l-1.68-8h1.232l1.296 6.24h.096l1.552-4.48h1.088zm14.55 8.016l-1.056.464-5.92-13.184 1.072-.448 5.904 13.168zm2.979-8.016l1.6 4.448h.112L180.7 4h1.168l-1.568 8h-1.424l-1.536-4.416h-.048l-1.6 4.416h-1.424l-1.68-8h1.232l1.296 6.24h.096l1.552-4.48h1.088zm14.55 8.016l-1.057.464-5.92-13.184 1.072-.448 5.904 13.168zm2.978-8.016l1.6 4.448h.112L198.23 4h1.168l-1.568 8h-1.424l-1.536-4.416h-.048l-1.6 4.416h-1.424l-1.68-8h1.232l1.296 6.24h.096l1.552-4.48h1.088zm14.55 8.016l-1.056.464-5.92-13.184 1.072-.448 5.904 13.168zM213.07.8h2.592v8.464c0 .096.005.216.016.36a17.83 17.83 0 00.096.928c.021.155.042.29.064.408h1.072V12h-2.128l-.16-1.232h-.064c-.235.405-.584.744-1.048 1.016-.464.272-1 .408-1.608.408-1.206 0-2.094-.339-2.664-1.016-.571-.677-.856-1.73-.856-3.16 0-.672.096-1.267.288-1.784.192-.517.466-.95.824-1.296.357-.347.786-.61 1.288-.792a4.912 4.912 0 011.68-.272c.224 0 .424.005.6.016.176.01.338.027.488.048.149.021.29.05.424.088l.44.12V1.872h-1.344V.8zm-.96 10.32c.65 0 1.162-.168 1.536-.504.373-.336.629-.84.768-1.512V5.392a2.478 2.478 0 00-.792-.352c-.294-.075-.68-.112-1.16-.112-.854 0-1.526.248-2.016.744-.491.496-.736 1.277-.736 2.344 0 .437.04.845.12 1.224.08.379.213.707.4.984.186.277.434.496.744.656.309.16.688.24 1.136.24z"/></g></g></svg> \ No newline at end of file diff --git a/9-regular-expressions/02-regexp-methods/article.md b/9-regular-expressions/02-regexp-methods/article.md deleted file mode 100644 index f66a9c794..000000000 --- a/9-regular-expressions/02-regexp-methods/article.md +++ /dev/null @@ -1,458 +0,0 @@ -# Methods of RegExp and String - -There are two sets of methods to deal with regular expressions. - -1. First, regular expressions are objects of the built-in [RegExp](mdn:js/RegExp) class, it provides many methods. -2. Besides that, there are methods in regular strings can work with regexps. - - -## Recipes - -Which method to use depends on what we'd like to do. - -Methods become much easier to understand if we separate them by their use in real-life tasks. - -So, here are general recipes, the details to follow: - -**To search for all matches:** - -Use regexp `g` flag and: -- Get a flat array of matches -- `str.match(reg)` -- Get an array or matches with details -- `str.matchAll(reg)`. - -**To search for the first match only:** -- Get the full first match -- `str.match(reg)` (without `g` flag). -- Get the string position of the first match -- `str.search(reg)`. -- Check if there's a match -- `regexp.test(str)`. -- Find the match from the given position -- `regexp.exec(str)` (set `regexp.lastIndex` to position). - -**To replace all matches:** -- Replace with another string or a function result -- `str.replace(reg, str|func)` - -**To split the string by a separator:** -- `str.split(str|reg)` - -Now you can continue reading this chapter to get the details about every method... But if you're reading for the first time, then you probably want to know more about regexps. So you can move to the next chapter, and then return here if something about a method is unclear. - -## str.search(reg) - -We've seen this method already. It returns the position of the first match or `-1` if none found: - -```js run -let str = "A drop of ink may make a million think"; - -alert( str.search( *!*/a/i*/!* ) ); // 0 (first match at zero position) -``` - -**The important limitation: `search` only finds the first match.** - -We can't find next matches using `search`, there's just no syntax for that. But there are other methods that can. - -## str.match(reg), no "g" flag - -The behavior of `str.match` varies depending on whether `reg` has `g` flag or not. - -First, if there's no `g` flag, then `str.match(reg)` looks for the first match only. - -The result is an array with that match and additional properties: - -- `index` -- the position of the match inside the string, -- `input` -- the subject string. - -For instance: - -```js run -let str = "Fame is the thirst of youth"; - -let result = str.match( *!*/fame/i*/!* ); - -alert( result[0] ); // Fame (the match) -alert( result.index ); // 0 (at the zero position) -alert( result.input ); // "Fame is the thirst of youth" (the string) -``` - -A match result may have more than one element. - -**If a part of the pattern is delimited by parentheses `(...)`, then it becomes a separate element in the array.** - -If parentheses have a name, designated by `(?<name>...)` at their start, then `result.groups[name]` has the content. We'll see that later in the chapter [about groups](info:regexp-groups). - -For instance: - -```js run -let str = "JavaScript is a programming language"; - -let result = str.match( *!*/JAVA(SCRIPT)/i*/!* ); - -alert( result[0] ); // JavaScript (the whole match) -alert( result[1] ); // script (the part of the match that corresponds to the parentheses) -alert( result.index ); // 0 -alert( result.input ); // JavaScript is a programming language -``` - -Due to the `i` flag the search is case-insensitive, so it finds `match:JavaScript`. The part of the match that corresponds to `pattern:SCRIPT` becomes a separate array item. - -So, this method is used to find one full match with all details. - - -## str.match(reg) with "g" flag - -When there's a `"g"` flag, then `str.match` returns an array of all matches. There are no additional properties in that array, and parentheses do not create any elements. - -For instance: - -```js run -let str = "HO-Ho-ho!"; - -let result = str.match( *!*/ho/ig*/!* ); - -alert( result ); // HO, Ho, ho (array of 3 matches, case-insensitive) -``` - -Parentheses do not change anything, here we go: - -```js run -let str = "HO-Ho-ho!"; - -let result = str.match( *!*/h(o)/ig*/!* ); - -alert( result ); // HO, Ho, ho -``` - -**So, with `g` flag `str.match` returns a simple array of all matches, without details.** - -If we want to get information about match positions and contents of parentheses then we should use `matchAll` method that we'll cover below. - -````warn header="If there are no matches, `str.match` returns `null`" -Please note, that's important. If there are no matches, the result is not an empty array, but `null`. - -Keep that in mind to evade pitfalls like this: - -```js run -let str = "Hey-hey-hey!"; - -alert( str.match(/Z/g).length ); // Error: Cannot read property 'length' of null -``` - -Here `str.match(/Z/g)` is `null`, it has no `length` property. -```` - -## str.matchAll(regexp) - -The method `str.matchAll(regexp)` is used to find all matches with all details. - -For instance: - -```js run -let str = "Javascript or JavaScript? Should we uppercase 'S'?"; - -let result = str.matchAll( *!*/java(script)/ig*/!* ); - -let [match1, match2] = result; - -alert( match1[0] ); // Javascript (the whole match) -alert( match1[1] ); // script (the part of the match that corresponds to the parentheses) -alert( match1.index ); // 0 -alert( match1.input ); // = str (the whole original string) - -alert( match2[0] ); // JavaScript (the whole match) -alert( match2[1] ); // Script (the part of the match that corresponds to the parentheses) -alert( match2.index ); // 14 -alert( match2.input ); // = str (the whole original string) -``` - -````warn header="`matchAll` returns an iterable, not array" -For instance, if we try to get the first match by index, it won't work: - -```js run -let str = "Javascript or JavaScript??"; - -let result = str.matchAll( /javascript/ig ); - -*!* -alert(result[0]); // undefined (?! there must be a match) -*/!* -``` - -The reason is that the iterator is not an array. We need to run `Array.from(result)` on it, or use `for..of` loop to get matches. - -In practice, if we need all matches, then `for..of` works, so it's not a problem. - -And, to get only few matches, we can use destructuring: - -```js run -let str = "Javascript or JavaScript??"; - -*!* -let [firstMatch] = str.matchAll( /javascript/ig ); -*/!* - -alert(firstMatch); // Javascript -``` -```` - -```warn header="`matchAll` is supernew, may need a polyfill" -The method may not work in old browsers. A polyfill might be needed (this site uses core-js). - -Or you could make a loop with `regexp.exec`, explained below. -``` - -## str.split(regexp|substr, limit) - -Splits the string using the regexp (or a substring) as a delimiter. - -We already used `split` with strings, like this: - -```js run -alert('12-34-56'.split('-')) // array of [12, 34, 56] -``` - -But we can split by a regular expression, the same way: - -```js run -alert('12-34-56'.split(/-/)) // array of [12, 34, 56] -``` - -## str.replace(str|reg, str|func) - -This is a generic method for searching and replacing, one of most useful ones. The swiss army knife for searching and replacing. - -We can use it without regexps, to search and replace a substring: - -```js run -// replace a dash by a colon -alert('12-34-56'.replace("-", ":")) // 12:34-56 -``` - -There's a pitfall though. - -**When the first argument of `replace` is a string, it only looks for the first match.** - -You can see that in the example above: only the first `"-"` is replaced by `":"`. - -To find all dashes, we need to use not the string `"-"`, but a regexp `pattern:/-/g`, with an obligatory `g` flag: - -```js run -// replace all dashes by a colon -alert( '12-34-56'.replace( *!*/-/g*/!*, ":" ) ) // 12:34:56 -``` - -The second argument is a replacement string. We can use special characters in it: - -| Symbol | Inserts | -|--------|--------| -|`$$`|`"$"` | -|`$&`|the whole match| -|<code>$`</code>|a part of the string before the match| -|`$'`|a part of the string after the match| -|`$n`|if `n` is a 1-2 digit number, then it means the contents of n-th parentheses counting from left to right, otherwise it means a parentheses with the given name | - - -For instance if we use `$&` in the replacement string, that means "put the whole match here". - -Let's use it to prepend all entries of `"John"` with `"Mr."`: - -```js run -let str = "John Doe, John Smith and John Bull"; - -// for each John - replace it with Mr. and then John -alert(str.replace(/John/g, 'Mr.$&')); // Mr.John Doe, Mr.John Smith and Mr.John Bull -``` - -Quite often we'd like to reuse parts of the source string, recombine them in the replacement or wrap into something. - -To do so, we should: -1. First, mark the parts by parentheses in regexp. -2. Use `$1`, `$2` (and so on) in the replacement string to get the content matched by 1st, 2nd and so on parentheses. - -For instance: - -```js run -let str = "John Smith"; - -// swap first and last name -alert(str.replace(/(john) (smith)/i, '$2, $1')) // Smith, John -``` - -**For situations that require "smart" replacements, the second argument can be a function.** - -It will be called for each match, and its result will be inserted as a replacement. - -For instance: - -```js run -let i = 0; - -// replace each "ho" by the result of the function -alert("HO-Ho-ho".replace(/ho/gi, function() { - return ++i; -})); // 1-2-3 -``` - -In the example above the function just returns the next number every time, but usually the result is based on the match. - -The function is called with arguments `func(str, p1, p2, ..., pn, offset, input, groups)`: - -1. `str` -- the match, -2. `p1, p2, ..., pn` -- contents of parentheses (if there are any), -3. `offset` -- position of the match, -4. `input` -- the source string, -5. `groups` -- an object with named groups (see chapter [](info:regexp-groups)). - -If there are no parentheses in the regexp, then there are only 3 arguments: `func(str, offset, input)`. - -Let's use it to show full information about matches: - -```js run -// show and replace all matches -function replacer(str, offset, input) { - alert(`Found ${str} at position ${offset} in string ${input}`); - return str.toLowerCase(); -} - -let result = "HO-Ho-ho".replace(/ho/gi, replacer); -alert( 'Result: ' + result ); // Result: ho-ho-ho - -// shows each match: -// Found HO at position 0 in string HO-Ho-ho -// Found Ho at position 3 in string HO-Ho-ho -// Found ho at position 6 in string HO-Ho-ho -``` - -In the example below there are two parentheses, so `replacer` is called with 5 arguments: `str` is the full match, then parentheses, and then `offset` and `input`: - -```js run -function replacer(str, name, surname, offset, input) { - // name is the first parentheses, surname is the second one - return surname + ", " + name; -} - -let str = "John Smith"; - -alert(str.replace(/(John) (Smith)/, replacer)) // Smith, John -``` - -Using a function gives us the ultimate replacement power, because it gets all the information about the match, has access to outer variables and can do everything. - -## regexp.exec(str) - -We've already seen these searching methods: - -- `search` -- looks for the position of the match, -- `match` -- if there's no `g` flag, returns the first match with parentheses and all details, -- `match` -- if there's a `g` flag -- returns all matches, without details parentheses, -- `matchAll` -- returns all matches with details. - -The `regexp.exec` method is the most flexible searching method of all. Unlike previous methods, `exec` should be called on a regexp, rather than on a string. - -It behaves differently depending on whether the regexp has the `g` flag. - -If there's no `g`, then `regexp.exec(str)` returns the first match, exactly as `str.match(reg)`. Such behavior does not give us anything new. - -But if there's `g`, then: -- `regexp.exec(str)` returns the first match and *remembers* the position after it in `regexp.lastIndex` property. -- The next call starts to search from `regexp.lastIndex` and returns the next match. -- If there are no more matches then `regexp.exec` returns `null` and `regexp.lastIndex` is set to `0`. - -We could use it to get all matches with their positions and parentheses groups in a loop, instead of `matchAll`: - -```js run -let str = 'A lot about JavaScript at https://javascript.info'; - -let regexp = /javascript/ig; - -let result; - -while (result = regexp.exec(str)) { - alert( `Found ${result[0]} at ${result.index}` ); - // shows: Found JavaScript at 12, then: - // shows: Found javascript at 34 -} -``` - -Surely, `matchAll` does the same, at least for modern browsers. But what `matchAll` can't do -- is to search from a given position. - -Let's search from position `13`. What we need is to assign `regexp.lastIndex=13` and call `regexp.exec`: - -```js run -let str = "A lot about JavaScript at https://javascript.info"; - -let regexp = /javascript/ig; -*!* -regexp.lastIndex = 13; -*/!* - -let result; - -while (result = regexp.exec(str)) { - alert( `Found ${result[0]} at ${result.index}` ); - // shows: Found javascript at 34 -} -``` - -Now, starting from the given position `13`, there's only one match. - - -## regexp.test(str) - -The method `regexp.test(str)` looks for a match and returns `true/false` whether it finds it. - -For instance: - -```js run -let str = "I love JavaScript"; - -// these two tests do the same -alert( *!*/love/i*/!*.test(str) ); // true -alert( str.search(*!*/love/i*/!*) != -1 ); // true -``` - -An example with the negative answer: - -```js run -let str = "Bla-bla-bla"; - -alert( *!*/love/i*/!*.test(str) ); // false -alert( str.search(*!*/love/i*/!*) != -1 ); // false -``` - -If the regexp has `'g'` flag, then `regexp.test` advances `regexp.lastIndex` property, just like `regexp.exec`. - -So we can use it to search from a given position: - -```js run -let regexp = /love/gi; - -let str = "I love JavaScript"; - -// start the search from position 10: -regexp.lastIndex = 10 -alert( regexp.test(str) ); // false (no match) -``` - - - -````warn header="Same global regexp tested repeatedly may fail to match" -If we apply the same global regexp to different inputs, it may lead to wrong result, because `regexp.test` call advances `regexp.lastIndex` property, so the search in another string may start from non-zero position. - -For instance, here we call `regexp.test` twice on the same text, and the second time fails: - -```js run -let regexp = /javascript/g; // (regexp just created: regexp.lastIndex=0) - -alert( regexp.test("javascript") ); // true (regexp.lastIndex=10 now) -alert( regexp.test("javascript") ); // false -``` - -That's exactly because `regexp.lastIndex` is non-zero on the second test. - -To work around that, one could use non-global regexps or re-adjust `regexp.lastIndex=0` before a new search. -```` - -## Summary - -There's a variety of many methods on both regexps and strings. - -Their abilities and methods overlap quite a bit, we can do the same by different calls. Sometimes that may cause confusion when starting to learn the language. - -Then please refer to the recipes at the beginning of this chapter, as they provide solutions for the majority of regexp-related tasks. diff --git a/9-regular-expressions/03-regexp-character-classes/hello-java-boundaries.svg b/9-regular-expressions/03-regexp-character-classes/hello-java-boundaries.svg deleted file mode 100644 index 65714ef75..000000000 --- a/9-regular-expressions/03-regexp-character-classes/hello-java-boundaries.svg +++ /dev/null @@ -1,18 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> -<svg width="245px" height="74px" viewBox="0 0 245 74" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> - <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> - <title>hello-java-boundaries.svg</title> - <desc>Created with sketchtool.</desc> - <g id="regexp" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> - <g id="hello-java-boundaries.svg"> - <text id="Hello,-Java!" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal" letter-spacing="4.58333333" fill="#8A704D"> - <tspan x="20" y="61">Hello, Java</tspan> - <tspan x="215.616667" y="61">!</tspan> - </text> - <path id="Line" d="M15.5,25.5 L15.5,8.5 L17.5,8.5 L17.5,25.5 L23.5,25.5 L16.5,39.5 L9.5,25.5 L15.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - <path id="Line-2" d="M110.5,25.5 L110.5,8.5 L112.5,8.5 L112.5,25.5 L118.5,25.5 L111.5,39.5 L104.5,25.5 L110.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - <path id="Line-3" d="M142.5,25.5 L142.5,8.5 L144.5,8.5 L144.5,25.5 L150.5,25.5 L143.5,39.5 L136.5,25.5 L142.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - <path id="Line-4" d="M216.5,25.5 L216.5,8.5 L218.5,8.5 L218.5,25.5 L224.5,25.5 L217.5,39.5 L210.5,25.5 L216.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - </g> - </g> -</svg> \ No newline at end of file diff --git a/9-regular-expressions/03-regexp-unicode/article.md b/9-regular-expressions/03-regexp-unicode/article.md new file mode 100644 index 000000000..1be163aff --- /dev/null +++ b/9-regular-expressions/03-regexp-unicode/article.md @@ -0,0 +1,161 @@ +# Unicode: flag "u" and class \p{...} + +JavaScript uses [Unicode encoding](https://en.wikipedia.org/wiki/Unicode) for strings. Most characters are encoded with 2 bytes, but that allows to represent at most 65536 characters. + +That range is not big enough to encode all possible characters, that's why some rare characters are encoded with 4 bytes, for instance like `𝒳` (mathematical X) or `😄` (a smile), some hieroglyphs and so on. + +Here are the Unicode values of some characters: + +| Character | Unicode | Bytes count in Unicode | +|------------|---------|--------| +| a | `0x0061` | 2 | +| ≈ | `0x2248` | 2 | +|𝒳| `0x1d4b3` | 4 | +|𝒴| `0x1d4b4` | 4 | +|😄| `0x1f604` | 4 | + +So characters like `a` and `≈` occupy 2 bytes, while codes for `𝒳`, `𝒴` and `😄` are longer, they have 4 bytes. + +Long time ago, when JavaScript language was created, Unicode encoding was simpler: there were no 4-byte characters. So, some language features still handle them incorrectly. + +For instance, `length` thinks that here are two characters: + +```js run +alert('😄'.length); // 2 +alert('𝒳'.length); // 2 +``` + +...But we can see that there's only one, right? The point is that `length` treats 4 bytes as two 2-byte characters. That's incorrect, because they must be considered only together (so-called "surrogate pair", you can read about them in the article <info:string>). + +By default, regular expressions also treat 4-byte "long characters" as a pair of 2-byte ones. And, as it happens with strings, that may lead to odd results. We'll see that a bit later, in the article <info:regexp-character-sets-and-ranges>. + +Unlike strings, regular expressions have flag `pattern:u` that fixes such problems. With such flag, a regexp handles 4-byte characters correctly. And also Unicode property search becomes available, we'll get to it next. + +## Unicode properties \p{...} + +Every character in Unicode has a lot of properties. They describe what "category" the character belongs to, contain miscellaneous information about it. + +For instance, if a character has `Letter` property, it means that the character belongs to an alphabet (of any language). And `Number` property means that it's a digit: maybe Arabic or Chinese, and so on. + +We can search for characters with a property, written as `pattern:\p{…}`. To use `pattern:\p{…}`, a regular expression must have flag `pattern:u`. + +For instance, `\p{Letter}` denotes a letter in any language. We can also use `\p{L}`, as `L` is an alias of `Letter`. There are shorter aliases for almost every property. + +In the example below three kinds of letters will be found: English, Georgian and Korean. + +```js run +let str = "A ბ ㄱ"; + +alert( str.match(/\p{L}/gu) ); // A,ბ,ㄱ +alert( str.match(/\p{L}/g) ); // null (no matches, \p doesn't work without the flag "u") +``` + +Here's the main character categories and their subcategories: + +- Letter `L`: + - lowercase `Ll` + - modifier `Lm`, + - titlecase `Lt`, + - uppercase `Lu`, + - other `Lo`. +- Number `N`: + - decimal digit `Nd`, + - letter number `Nl`, + - other `No`. +- Punctuation `P`: + - connector `Pc`, + - dash `Pd`, + - initial quote `Pi`, + - final quote `Pf`, + - open `Ps`, + - close `Pe`, + - other `Po`. +- Mark `M` (accents etc): + - spacing combining `Mc`, + - enclosing `Me`, + - non-spacing `Mn`. +- Symbol `S`: + - currency `Sc`, + - modifier `Sk`, + - math `Sm`, + - other `So`. +- Separator `Z`: + - line `Zl`, + - paragraph `Zp`, + - space `Zs`. +- Other `C`: + - control `Cc`, + - format `Cf`, + - not assigned `Cn`, + - private use `Co`, + - surrogate `Cs`. + + +So, e.g. if we need letters in lower case, we can write `pattern:\p{Ll}`, punctuation signs: `pattern:\p{P}` and so on. + +There are also other derived categories, like: +- `Alphabetic` (`Alpha`), includes Letters `L`, plus letter numbers `Nl` (e.g. Ⅻ - a character for the roman number 12), plus some other symbols `Other_Alphabetic` (`OAlpha`). +- `Hex_Digit` includes hexadecimal digits: `0-9`, `a-f`. +- ...And so on. + +Unicode supports many different properties, their full list would require a lot of space, so here are the references: + +- List all properties by a character: <https://unicode.org/cldr/utility/character.jsp>. +- List all characters by a property: <https://unicode.org/cldr/utility/list-unicodeset.jsp>. +- Short aliases for properties: <https://www.unicode.org/Public/UCD/latest/ucd/PropertyValueAliases.txt>. +- A full base of Unicode characters in text format, with all properties, is here: <https://www.unicode.org/Public/UCD/latest/ucd/>. + +### Example: hexadecimal numbers + +For instance, let's look for hexadecimal numbers, written as `xFF`, where `F` is a hex digit (0..9 or A..F). + +A hex digit can be denoted as `pattern:\p{Hex_Digit}`: + +```js run +let regexp = /x\p{Hex_Digit}\p{Hex_Digit}/u; + +alert("number: xAF".match(regexp)); // xAF +``` + +### Example: Chinese hieroglyphs + +Let's look for Chinese hieroglyphs. + +There's a Unicode property `Script` (a writing system), that may have a value: `Cyrillic`, `Greek`, `Arabic`, `Han` (Chinese) and so on, [here's the full list](https://en.wikipedia.org/wiki/Script_(Unicode)). + +To look for characters in a given writing system we should use `pattern:Script=<value>`, e.g. for Cyrillic letters: `pattern:\p{sc=Cyrillic}`, for Chinese hieroglyphs: `pattern:\p{sc=Han}`, and so on: + +```js run +let regexp = /\p{sc=Han}/gu; // returns Chinese hieroglyphs + +let str = `Hello Привет 你好 123_456`; + +alert( str.match(regexp) ); // 你,好 +``` + +### Example: currency + +Characters that denote a currency, such as `$`, `€`, `¥`, have Unicode property `pattern:\p{Currency_Symbol}`, the short alias: `pattern:\p{Sc}`. + +Let's use it to look for prices in the format "currency, followed by a digit": + +```js run +let regexp = /\p{Sc}\d/gu; + +let str = `Prices: $2, €1, ¥9`; + +alert( str.match(regexp) ); // $2,€1,¥9 +``` + +Later, in the article <info:regexp-quantifiers> we'll see how to look for numbers that contain many digits. + +## Summary + +Flag `pattern:u` enables the support of Unicode in regular expressions. + +That means two things: + +1. Characters of 4 bytes are handled correctly: as a single character, not two 2-byte characters. +2. Unicode properties can be used in the search: `\p{…}`. + +With Unicode properties we can look for words in given languages, special characters (quotes, currencies) and so on. diff --git a/9-regular-expressions/12-regexp-anchors/1-start-end/solution.md b/9-regular-expressions/04-regexp-anchors/1-start-end/solution.md similarity index 77% rename from 9-regular-expressions/12-regexp-anchors/1-start-end/solution.md rename to 9-regular-expressions/04-regexp-anchors/1-start-end/solution.md index 1a8cbe9a2..702f992d7 100644 --- a/9-regular-expressions/12-regexp-anchors/1-start-end/solution.md +++ b/9-regular-expressions/04-regexp-anchors/1-start-end/solution.md @@ -1,5 +1,4 @@ - -The empty string is the only match: it starts and immediately finishes. +An empty string is the only match: it starts and immediately finishes. The task once again demonstrates that anchors are not characters, but tests. diff --git a/9-regular-expressions/12-regexp-anchors/1-start-end/task.md b/9-regular-expressions/04-regexp-anchors/1-start-end/task.md similarity index 100% rename from 9-regular-expressions/12-regexp-anchors/1-start-end/task.md rename to 9-regular-expressions/04-regexp-anchors/1-start-end/task.md diff --git a/9-regular-expressions/04-regexp-anchors/article.md b/9-regular-expressions/04-regexp-anchors/article.md new file mode 100644 index 000000000..63e4efa93 --- /dev/null +++ b/9-regular-expressions/04-regexp-anchors/article.md @@ -0,0 +1,52 @@ +# Anchors: string start ^ and end $ + +The caret `pattern:^` and dollar `pattern:$` characters have special meaning in a regexp. They are called "anchors". + +The caret `pattern:^` matches at the beginning of the text, and the dollar `pattern:$` -- at the end. + +For instance, let's test if the text starts with `Mary`: + +```js run +let str1 = "Mary had a little lamb"; +alert( /^Mary/.test(str1) ); // true +``` + +The pattern `pattern:^Mary` means: "string start and then Mary". + +Similar to this, we can test if the string ends with `snow` using `pattern:snow$`: + +```js run +let str1 = "its fleece was white as snow"; +alert( /snow$/.test(str1) ); // true +``` + +In these particular cases we could use string methods `startsWith/endsWith` instead. Regular expressions should be used for more complex tests. + +## Testing for a full match + +Both anchors together `pattern:^...$` are often used to test whether or not a string fully matches the pattern. For instance, to check if the user input is in the right format. + +Let's check whether or not a string is a time in `12:34` format. That is: two digits, then a colon, and then another two digits. + +In regular expressions language that's `pattern:\d\d:\d\d`: + +```js run +let goodInput = "12:34"; +let badInput = "12:345"; + +let regexp = /^\d\d:\d\d$/; +alert( regexp.test(goodInput) ); // true +alert( regexp.test(badInput) ); // false +``` + +Here the match for `pattern:\d\d:\d\d` must start exactly after the beginning of the text `pattern:^`, and the end `pattern:$` must immediately follow. + +The whole string must be exactly in this format. If there's any deviation or an extra character, the result is `false`. + +Anchors behave differently if flag `pattern:m` is present. We'll see that in the next article. + +```smart header="Anchors have \"zero width\"" +Anchors `pattern:^` and `pattern:$` are tests. They have zero width. + +In other words, they do not match a character, but rather force the regexp engine to check the condition (text start/end). +``` diff --git a/9-regular-expressions/04-regexp-escaping/article.md b/9-regular-expressions/04-regexp-escaping/article.md deleted file mode 100644 index 48f4a810d..000000000 --- a/9-regular-expressions/04-regexp-escaping/article.md +++ /dev/null @@ -1,99 +0,0 @@ - -# Escaping, special characters - -As we've seen, a backslash `"\"` is used to denote character classes. So it's a special character in regexps (just like in a regular string). - -There are other special characters as well, that have special meaning in a regexp. They are used to do more powerful searches. Here's a full list of them: `pattern:[ \ ^ $ . | ? * + ( )`. - -Don't try to remember the list -- soon we'll deal with each of them separately and you'll know them by heart automatically. - -## Escaping - -Let's say we want to find a dot literally. Not "any character", but just a dot. - -To use a special character as a regular one, prepend it with a backslash: `pattern:\.`. - -That's also called "escaping a character". - -For example: -```js run -alert( "Chapter 5.1".match(/\d\.\d/) ); // 5.1 (match!) -alert( "Chapter 511".match(/\d\.\d/) ); // null (looking for a real dot \.) -``` - -Parentheses are also special characters, so if we want them, we should use `pattern:\(`. The example below looks for a string `"g()"`: - -```js run -alert( "function g()".match(/g\(\)/) ); // "g()" -``` - -If we're looking for a backslash `\`, it's a special character in both regular strings and regexps, so we should double it. - -```js run -alert( "1\\2".match(/\\/) ); // '\' -``` - -## A slash - -A slash symbol `'/'` is not a special character, but in JavaScript it is used to open and close the regexp: `pattern:/...pattern.../`, so we should escape it too. - -Here's what a search for a slash `'/'` looks like: - -```js run -alert( "/".match(/\//) ); // '/' -``` - -On the other hand, if we're not using `/.../`, but create a regexp using `new RegExp`, then we don't need to escape it: - -```js run -alert( "/".match(new RegExp("/")) ); // '/' -``` - -## new RegExp - -If we are creating a regular expression with `new RegExp`, then we don't have to escape `/`, but need to do some other escaping. - -For instance, consider this: - -```js run -let reg = new RegExp("\d\.\d"); - -alert( "Chapter 5.1".match(reg) ); // null -``` - -It worked with `pattern:/\d\.\d/`, but with `new RegExp("\d\.\d")` it doesn't, why? - -The reason is that backslashes are "consumed" by a string. Remember, regular strings have their own special characters like `\n`, and a backslash is used for escaping. - -Please, take a look, what "\d\.\d" really is: - -```js run -alert("\d\.\d"); // d.d -``` - -The quotes "consume" backslashes and interpret them, for instance: - -- `\n` -- becomes a newline character, -- `\u1234` -- becomes the Unicode character with such code, -- ...And when there's no special meaning: like `\d` or `\z`, then the backslash is simply removed. - -So the call to `new RegExp` gets a string without backslashes. That's why it doesn't work! - -To fix it, we need to double backslashes, because quotes turn `\\` into `\`: - -```js run -*!* -let regStr = "\\d\\.\\d"; -*/!* -alert(regStr); // \d\.\d (correct now) - -let reg = new RegExp(regStr); - -alert( "Chapter 5.1".match(reg) ); // 5.1 -``` - -## Summary - -- To search special characters `pattern:[ \ ^ $ . | ? * + ( )` literally, we need to prepend them with `\` ("escape them"). -- We also need to escape `/` if we're inside `pattern:/.../` (but not inside `new RegExp`). -- When passing a string `new RegExp`, we need to double backslashes `\\`, cause strings consume one of them. diff --git a/9-regular-expressions/05-regexp-character-sets-and-ranges/article.md b/9-regular-expressions/05-regexp-character-sets-and-ranges/article.md deleted file mode 100644 index 5c8a8babb..000000000 --- a/9-regular-expressions/05-regexp-character-sets-and-ranges/article.md +++ /dev/null @@ -1,114 +0,0 @@ -# Sets and ranges [...] - -Several characters or character classes inside square brackets `[…]` mean to "search for any character among given". - -## Sets - -For instance, `pattern:[eao]` means any of the 3 characters: `'a'`, `'e'`, or `'o'`. - -That's called a *set*. Sets can be used in a regexp along with regular characters: - -```js run -// find [t or m], and then "op" -alert( "Mop top".match(/[tm]op/gi) ); // "Mop", "top" -``` - -Please note that although there are multiple characters in the set, they correspond to exactly one character in the match. - -So the example above gives no matches: - -```js run -// find "V", then [o or i], then "la" -alert( "Voila".match(/V[oi]la/) ); // null, no matches -``` - -The pattern assumes: - -- `pattern:V`, -- then *one* of the letters `pattern:[oi]`, -- then `pattern:la`. - -So there would be a match for `match:Vola` or `match:Vila`. - -## Ranges - -Square brackets may also contain *character ranges*. - -For instance, `pattern:[a-z]` is a character in range from `a` to `z`, and `pattern:[0-5]` is a digit from `0` to `5`. - -In the example below we're searching for `"x"` followed by two digits or letters from `A` to `F`: - -```js run -alert( "Exception 0xAF".match(/x[0-9A-F][0-9A-F]/g) ); // xAF -``` - -Please note that in the word `subject:Exception` there's a substring `subject:xce`. It didn't match the pattern, because the letters are lowercase, while in the set `pattern:[0-9A-F]` they are uppercase. - -If we want to find it too, then we can add a range `a-f`: `pattern:[0-9A-Fa-f]`. The `i` flag would allow lowercase too. - -**Character classes are shorthands for certain character sets.** - -For instance: - -- **\d** -- is the same as `pattern:[0-9]`, -- **\w** -- is the same as `pattern:[a-zA-Z0-9_]`, -- **\s** -- is the same as `pattern:[\t\n\v\f\r ]` plus few other unicode space characters. - -We can use character classes inside `[…]` as well. - -For instance, we want to match all wordly characters or a dash, for words like "twenty-third". We can't do it with `pattern:\w+`, because `pattern:\w` class does not include a dash. But we can use `pattern:[\w-]`. - -We also can use a combination of classes to cover every possible character, like `pattern:[\s\S]`. That matches spaces or non-spaces -- any character. That's wider than a dot `"."`, because the dot matches any character except a newline. - -## Excluding ranges - -Besides normal ranges, there are "excluding" ranges that look like `pattern:[^…]`. - -They are denoted by a caret character `^` at the start and match any character *except the given ones*. - -For instance: - -- `pattern:[^aeyo]` -- any character except `'a'`, `'e'`, `'y'` or `'o'`. -- `pattern:[^0-9]` -- any character except a digit, the same as `\D`. -- `pattern:[^\s]` -- any non-space character, same as `\S`. - -The example below looks for any characters except letters, digits and spaces: - -```js run -alert( "alice15@gmail.com".match(/[^\d\sA-Z]/gi) ); // @ and . -``` - -## No escaping in […] - -Usually when we want to find exactly the dot character, we need to escape it like `pattern:\.`. And if we need a backslash, then we use `pattern:\\`. - -In square brackets the vast majority of special characters can be used without escaping: - -- A dot `pattern:'.'`. -- A plus `pattern:'+'`. -- Parentheses `pattern:'( )'`. -- Dash `pattern:'-'` in the beginning or the end (where it does not define a range). -- A caret `pattern:'^'` if not in the beginning (where it means exclusion). -- And the opening square bracket `pattern:'['`. - -In other words, all special characters are allowed except where they mean something for square brackets. - -A dot `"."` inside square brackets means just a dot. The pattern `pattern:[.,]` would look for one of characters: either a dot or a comma. - -In the example below the regexp `pattern:[-().^+]` looks for one of the characters `-().^+`: - -```js run -// No need to escape -let reg = /[-().^+]/g; - -alert( "1 + 2 - 3".match(reg) ); // Matches +, - -``` - -...But if you decide to escape them "just in case", then there would be no harm: - -```js run -// Escaped everything -let reg = /[\-\(\)\.\^\+]/g; - -alert( "1 + 2 - 3".match(reg) ); // also works: +, - -``` diff --git a/9-regular-expressions/05-regexp-multiline-mode/article.md b/9-regular-expressions/05-regexp-multiline-mode/article.md new file mode 100644 index 000000000..82f4d781d --- /dev/null +++ b/9-regular-expressions/05-regexp-multiline-mode/article.md @@ -0,0 +1,87 @@ +# Multiline mode of anchors ^ $, flag "m" + +The multiline mode is enabled by the flag `pattern:m`. + +It only affects the behavior of `pattern:^` and `pattern:$`. + +In the multiline mode they match not only at the beginning and the end of the string, but also at start/end of line. + +## Searching at line start ^ + +In the example below the text has multiple lines. The pattern `pattern:/^\d/gm` takes a digit from the beginning of each line: + +```js run +let str = `1st place: Winnie +2nd place: Piglet +3rd place: Eeyore`; + +*!* +console.log( str.match(/^\d/gm) ); // 1, 2, 3 +*/!* +``` + +Without the flag `pattern:m` only the first digit is matched: + +```js run +let str = `1st place: Winnie +2nd place: Piglet +3rd place: Eeyore`; + +*!* +console.log( str.match(/^\d/g) ); // 1 +*/!* +``` + +That's because by default a caret `pattern:^` only matches at the beginning of the text, and in the multiline mode -- at the start of any line. + +```smart +"Start of a line" formally means "immediately after a line break": the test `pattern:^` in multiline mode matches at all positions preceded by a newline character `\n`. + +And at the text start. +``` + +## Searching at line end $ + +The dollar sign `pattern:$` behaves similarly. + +The regular expression `pattern:\d$` finds the last digit in every line + +```js run +let str = `Winnie: 1 +Piglet: 2 +Eeyore: 3`; + +console.log( str.match(/\d$/gm) ); // 1,2,3 +``` + +Without the flag `pattern:m`, the dollar `pattern:$` would only match the end of the whole text, so only the very last digit would be found. + +```smart +"End of a line" formally means "immediately before a line break": the test `pattern:$` in multiline mode matches at all positions succeeded by a newline character `\n`. + +And at the text end. +``` + +## Searching for \n instead of ^ $ + +To find a newline, we can use not only anchors `pattern:^` and `pattern:$`, but also the newline character `\n`. + +What's the difference? Let's see an example. + +Here we search for `pattern:\d\n` instead of `pattern:\d$`: + +```js run +let str = `Winnie: 1 +Piglet: 2 +Eeyore: 3`; + +console.log( str.match(/\d\n/g) ); // 1\n,2\n +``` + +As we can see, there are 2 matches instead of 3. + +That's because there's no newline after `subject:3` (there's text end though, so it matches `pattern:$`). + +Another difference: now every match includes a newline character `match:\n`. Unlike the anchors `pattern:^` `pattern:$`, that only test the condition (start/end of a line), `\n` is a character, so it becomes a part of the result. + +So, a `\n` in the pattern is used when we need newline characters in the result, while anchors are used to find something at the beginning/end of a line. diff --git a/9-regular-expressions/03-regexp-character-classes/1-find-time-hh-mm/solution.md b/9-regular-expressions/06-regexp-boundary/1-find-time-hh-mm/solution.md similarity index 100% rename from 9-regular-expressions/03-regexp-character-classes/1-find-time-hh-mm/solution.md rename to 9-regular-expressions/06-regexp-boundary/1-find-time-hh-mm/solution.md diff --git a/9-regular-expressions/03-regexp-character-classes/1-find-time-hh-mm/task.md b/9-regular-expressions/06-regexp-boundary/1-find-time-hh-mm/task.md similarity index 99% rename from 9-regular-expressions/03-regexp-character-classes/1-find-time-hh-mm/task.md rename to 9-regular-expressions/06-regexp-boundary/1-find-time-hh-mm/task.md index 5e32b9c48..95ab5777d 100644 --- a/9-regular-expressions/03-regexp-character-classes/1-find-time-hh-mm/task.md +++ b/9-regular-expressions/06-regexp-boundary/1-find-time-hh-mm/task.md @@ -5,4 +5,5 @@ The time has a format: `hours:minutes`. Both hours and minutes has two digits, l Make a regexp to find time in the string: `subject:Breakfast at 09:00 in the room 123:456.` P.S. In this task there's no need to check time correctness yet, so `25:99` can also be a valid result. + P.P.S. The regexp shouldn't match `123:456`. diff --git a/9-regular-expressions/06-regexp-boundary/article.md b/9-regular-expressions/06-regexp-boundary/article.md new file mode 100644 index 000000000..06b5ac9f7 --- /dev/null +++ b/9-regular-expressions/06-regexp-boundary/article.md @@ -0,0 +1,52 @@ +# Word boundary: \b + +A word boundary `pattern:\b` is a test, just like `pattern:^` and `pattern:$`. + +When the regexp engine (program module that implements searching for regexps) comes across `pattern:\b`, it checks that the position in the string is a word boundary. + +There are three different positions that qualify as word boundaries: + +- At string start, if the first string character is a word character `pattern:\w`. +- Between two characters in the string, where one is a word character `pattern:\w` and the other is not. +- At string end, if the last string character is a word character `pattern:\w`. + +For instance, regexp `pattern:\bJava\b` will be found in `subject:Hello, Java!`, where `subject:Java` is a standalone word, but not in `subject:Hello, JavaScript!`. + +```js run +alert( "Hello, Java!".match(/\bJava\b/) ); // Java +alert( "Hello, JavaScript!".match(/\bJava\b/) ); // null +``` + +In the string `subject:Hello, Java!` following positions correspond to `pattern:\b`: + + + +So, it matches the pattern `pattern:\bHello\b`, because: + +1. At the beginning of the string matches the first test `pattern:\b`. +2. Then matches the word `pattern:Hello`. +3. Then the test `pattern:\b` matches again, as we're between `subject:o` and a comma. + +So the pattern `pattern:\bHello\b` would match, but not `pattern:\bHell\b` (because there's no word boundary after `l`) and not `Java!\b` (because the exclamation sign is not a wordly character `pattern:\w`, so there's no word boundary after it). + +```js run +alert( "Hello, Java!".match(/\bHello\b/) ); // Hello +alert( "Hello, Java!".match(/\bJava\b/) ); // Java +alert( "Hello, Java!".match(/\bHell\b/) ); // null (no match) +alert( "Hello, Java!".match(/\bJava!\b/) ); // null (no match) +``` + +We can use `pattern:\b` not only with words, but with digits as well. + +For example, the pattern `pattern:\b\d\d\b` looks for standalone 2-digit numbers. In other words, it looks for 2-digit numbers that are surrounded by characters different from `pattern:\w`, such as spaces or punctuation (or text start/end). + +```js run +alert( "1 23 456 78".match(/\b\d\d\b/g) ); // 23,78 +alert( "12,34,56".match(/\b\d\d\b/g) ); // 12,34,56 +``` + +```warn header="Word boundary `pattern:\b` doesn't work for non-latin alphabets" +The word boundary test `pattern:\b` checks that there should be `pattern:\w` on the one side from the position and "not `pattern:\w`" - on the other side. + +But `pattern:\w` means a latin letter `a-z` (or a digit or an underscore), so the test doesn't work for other characters, e.g. cyrillic letters or hieroglyphs. +``` diff --git a/9-regular-expressions/06-regexp-boundary/hello-java-boundaries.svg b/9-regular-expressions/06-regexp-boundary/hello-java-boundaries.svg new file mode 100644 index 000000000..0314308dd --- /dev/null +++ b/9-regular-expressions/06-regexp-boundary/hello-java-boundaries.svg @@ -0,0 +1,22 @@ +<<<<<<< HEAD:9-regular-expressions/03-regexp-character-classes/hello-java-boundaries.svg +<?xml version="1.0" encoding="UTF-8"?> +<svg width="245px" height="74px" viewBox="0 0 245 74" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> + <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> + <title>hello-java-boundaries.svg</title> + <desc>Created with sketchtool.</desc> + <g id="regexp" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> + <g id="hello-java-boundaries.svg"> + <text id="Hello,-Java!" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal" letter-spacing="4.58333333" fill="#8A704D"> + <tspan x="20" y="61">Hello, Java</tspan> + <tspan x="215.616667" y="61">!</tspan> + </text> + <path id="Line" d="M15.5,25.5 L15.5,8.5 L17.5,8.5 L17.5,25.5 L23.5,25.5 L16.5,39.5 L9.5,25.5 L15.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <path id="Line-2" d="M110.5,25.5 L110.5,8.5 L112.5,8.5 L112.5,25.5 L118.5,25.5 L111.5,39.5 L104.5,25.5 L110.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <path id="Line-3" d="M142.5,25.5 L142.5,8.5 L144.5,8.5 L144.5,25.5 L150.5,25.5 L143.5,39.5 L136.5,25.5 L142.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + <path id="Line-4" d="M216.5,25.5 L216.5,8.5 L218.5,8.5 L218.5,25.5 L224.5,25.5 L217.5,39.5 L210.5,25.5 L216.5,25.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + </g> + </g> +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="245" height="74" viewBox="0 0 245 74"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="hello-java-boundaries.svg" fill-rule="nonzero"><path id="Hello,Java!" fill="#AF6E24" d="M30.054 53.938h-6.908V61H21.32V45.6h1.826v6.71h6.908V45.6h1.826V61h-1.826v-7.062zm19.367 5.676a5.537 5.537 0 01-.935.66 6.9 6.9 0 01-1.177.528c-.425.147-.869.26-1.33.341-.463.08-.928.121-1.398.121-.88 0-1.657-.136-2.332-.407a4.49 4.49 0 01-1.694-1.166 5.081 5.081 0 01-1.034-1.815c-.234-.704-.352-1.496-.352-2.376 0-.924.129-1.745.385-2.464.257-.719.627-1.32 1.111-1.804a4.839 4.839 0 011.76-1.111c.69-.257 1.46-.385 2.31-.385.616 0 1.225.08 1.826.242a3.606 3.606 0 011.595.913c.462.447.818 1.063 1.067 1.848.25.785.338 1.8.264 3.047h-8.514c0 1.32.356 2.306 1.067 2.959.712.653 1.661.979 2.85.979.395 0 .788-.048 1.176-.143a9.833 9.833 0 001.111-.341c.352-.132.664-.279.935-.44.272-.161.48-.308.627-.44l.682 1.254zm-4.642-8.426a5.94 5.94 0 00-1.375.154 3.134 3.134 0 00-1.144.517c-.33.242-.601.557-.814.946-.212.389-.348.869-.407 1.441h6.842c-.073-.968-.385-1.72-.935-2.255-.55-.535-1.272-.803-2.167-.803zM58.185 45.6h3.652v11.704c0 .895.15 1.525.45 1.892.301.367.745.55 1.332.55.41 0 .803-.073 1.177-.22.374-.147.788-.396 1.243-.748l.792 1.21a4.874 4.874 0 01-.77.55 5.717 5.717 0 01-1.738.66 4.17 4.17 0 01-.814.088c-.543 0-1.027-.07-1.452-.209a2.481 2.481 0 01-1.067-.671c-.286-.308-.503-.715-.65-1.221-.146-.506-.22-1.133-.22-1.881v-10.23h-1.935V45.6zm17.783 0h3.652v11.704c0 .895.15 1.525.451 1.892.3.367.744.55 1.331.55.41 0 .803-.073 1.177-.22.374-.147.788-.396 1.243-.748l.792 1.21a4.874 4.874 0 01-.77.55 5.717 5.717 0 01-1.738.66 4.17 4.17 0 01-.814.088c-.543 0-1.027-.07-1.452-.209a2.481 2.481 0 01-1.067-.671c-.286-.308-.502-.715-.649-1.221-.147-.506-.22-1.133-.22-1.881v-10.23h-1.936V45.6zm16.42 9.9c0-1.775.461-3.179 1.385-4.213.924-1.034 2.244-1.551 3.96-1.551.924 0 1.72.15 2.387.451.668.3 1.221.708 1.661 1.221.44.513.767 1.122.98 1.826.212.704.318 1.46.318 2.266 0 .88-.117 1.676-.352 2.387a4.959 4.959 0 01-1.034 1.815 4.569 4.569 0 01-1.683 1.155c-.667.271-1.426.407-2.277.407-.909 0-1.701-.15-2.376-.451a4.663 4.663 0 01-1.672-1.221 4.979 4.979 0 01-.979-1.826 7.799 7.799 0 01-.319-2.266zm1.825 0c0 .513.063 1.027.187 1.54.125.513.327.975.605 1.386.279.41.642.74 1.09.99.447.25.993.374 1.638.374 1.174 0 2.057-.363 2.651-1.089.594-.726.891-1.793.891-3.201a6.46 6.46 0 00-.187-1.551 4.098 4.098 0 00-.616-1.375 3.247 3.247 0 00-1.1-.99c-.447-.25-.993-.374-1.639-.374-1.173 0-2.053.36-2.64 1.078-.586.719-.88 1.79-.88 3.212zm19.742 4.29c0-.44.143-.796.429-1.067.286-.271.656-.407 1.11-.407.558 0 .99.194 1.299.583.308.389.462.935.462 1.639 0 .572-.096 1.074-.286 1.507a3.802 3.802 0 01-1.64 1.826c-.33.183-.634.319-.912.407l-.55-.858a3.904 3.904 0 001.364-.88c.381-.381.572-.843.572-1.386a2.463 2.463 0 01-.484.066c-.426 0-.76-.121-1.001-.363-.242-.242-.363-.598-.363-1.067zm32.662-14.19h8.712v9.944c0 1.848-.44 3.256-1.32 4.224-.88.968-2.148 1.452-3.806 1.452a7.12 7.12 0 01-1.254-.11 7.234 7.234 0 01-1.144-.297 5.646 5.646 0 01-.913-.407c-.256-.147-.443-.286-.56-.418l.791-1.452c.162.103.341.213.54.33.197.117.417.231.66.341a4.331 4.331 0 001.837.385c1.1 0 1.932-.345 2.496-1.034.565-.69.847-1.804.847-3.344v-7.986h-6.886V45.6zm18.026 5.258a6.988 6.988 0 012.134-.792c.792-.161 1.569-.242 2.332-.242.748 0 1.36.117 1.837.352.476.235.85.539 1.122.913.271.374.454.8.55 1.276.095.477.143.957.143 1.441 0 .557-.015 1.151-.044 1.782a81.5 81.5 0 00-.066 1.892c0 .733.044 1.43.132 2.09h1.474V61h-2.926l-.198-1.65h-.11c-.088.132-.22.3-.396.506a3.623 3.623 0 01-.693.605c-.286.198-.635.37-1.045.517-.411.147-.895.22-1.452.22-1.086 0-1.944-.279-2.574-.836-.631-.557-.946-1.32-.946-2.288 0-.748.165-1.371.495-1.87.33-.499.803-.88 1.419-1.144.616-.264 1.36-.418 2.233-.462.872-.044 1.851.015 2.937.176.073-.675.084-1.236.033-1.683-.052-.447-.169-.803-.352-1.067a1.474 1.474 0 00-.77-.561c-.33-.11-.737-.165-1.221-.165-.66 0-1.291.092-1.892.275-.602.183-1.137.37-1.606.561l-.55-1.276zm3.234 8.866c.41 0 .792-.066 1.144-.198.352-.132.66-.3.924-.506.264-.205.484-.429.66-.671.176-.242.308-.473.396-.693v-1.54a12.447 12.447 0 00-2.112-.198c-.646 0-1.203.07-1.672.209-.47.14-.836.356-1.1.649-.264.293-.396.675-.396 1.144 0 .484.165.906.495 1.265.33.36.883.539 1.66.539zm18.817-.858h.242L190.39 50h1.914l-4.642 11h-1.936l-4.73-11h2.024l3.674 8.866zm13.515-8.008a6.988 6.988 0 012.134-.792c.792-.161 1.57-.242 2.332-.242.748 0 1.36.117 1.837.352.477.235.851.539 1.122.913.272.374.455.8.55 1.276.096.477.143.957.143 1.441 0 .557-.014 1.151-.044 1.782-.029.63-.051 1.261-.066 1.892 0 .733.044 1.43.132 2.09h1.474V61h-2.926l-.198-1.65h-.11c-.088.132-.22.3-.396.506a3.623 3.623 0 01-.693.605c-.286.198-.634.37-1.045.517-.41.147-.894.22-1.452.22-1.085 0-1.943-.279-2.574-.836-.63-.557-.946-1.32-.946-2.288 0-.748.165-1.371.495-1.87.33-.499.803-.88 1.42-1.144.615-.264 1.36-.418 2.232-.462.873-.044 1.852.015 2.937.176.074-.675.085-1.236.033-1.683-.051-.447-.168-.803-.352-1.067a1.474 1.474 0 00-.77-.561c-.33-.11-.737-.165-1.22-.165-.66 0-1.291.092-1.893.275-.601.183-1.136.37-1.606.561l-.55-1.276zm3.234 8.866c.411 0 .792-.066 1.144-.198.352-.132.66-.3.924-.506.264-.205.484-.429.66-.671.176-.242.308-.473.396-.693v-1.54a12.447 12.447 0 00-2.112-.198c-.645 0-1.202.07-1.672.209-.469.14-.836.356-1.1.649-.264.293-.396.675-.396 1.144 0 .484.165.906.495 1.265.33.36.884.539 1.661.539zM221.315 45.6h1.826v7.634l-.374 3.872h-1.078l-.374-3.872V45.6zm-.374 14.388c0-.396.113-.708.34-.935.228-.227.532-.341.914-.341.396 0 .711.114.946.341.234.227.352.539.352.935 0 .381-.118.69-.352.924-.235.235-.55.352-.946.352-.382 0-.686-.117-.913-.352-.228-.235-.341-.543-.341-.924z"/><path id="Line" fill="#C06334" d="M17.5 8.5v17h6l-7 14-7-14h6v-17h2z"/><path id="Line-2" fill="#C06334" d="M112.5 8.5v17h6l-7 14-7-14h6v-17h2z"/><path id="Line-3" fill="#C06334" d="M144.5 8.5v17h6l-7 14-7-14h6v-17h2z"/><path id="Line-4" fill="#C06334" d="M218.5 8.5v17h6l-7 14-7-14h6v-17h2z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/06-regexp-boundary/hello-java-boundaries.svg diff --git a/9-regular-expressions/03-regexp-character-classes/love-html5-classes.svg b/9-regular-expressions/06-regexp-boundary/love-html5-classes.svg similarity index 100% rename from 9-regular-expressions/03-regexp-character-classes/love-html5-classes.svg rename to 9-regular-expressions/06-regexp-boundary/love-html5-classes.svg diff --git a/9-regular-expressions/07-regexp-escaping/article.md b/9-regular-expressions/07-regexp-escaping/article.md new file mode 100644 index 000000000..1ad3ac98c --- /dev/null +++ b/9-regular-expressions/07-regexp-escaping/article.md @@ -0,0 +1,99 @@ + +# Escaping, special characters + +As we've seen, a backslash `pattern:\` is used to denote character classes, e.g. `pattern:\d`. So it's a special character in regexps (just like in regular strings). + +There are other special characters as well, that have special meaning in a regexp, such as `pattern:[ ] { } ( ) \ ^ $ . | ? * +`. They are used to do more powerful searches. + +Don't try to remember the list -- soon we'll deal with each of them, and you'll know them by heart automatically. + +## Escaping + +Let's say we want to find literally a dot. Not "any character", but just a dot. + +To use a special character as a regular one, prepend it with a backslash: `pattern:\.`. + +That's also called "escaping a character". + +For example: +```js run +alert( "Chapter 5.1".match(/\d\.\d/) ); // 5.1 (match!) +alert( "Chapter 511".match(/\d\.\d/) ); // null (looking for a real dot \.) +``` + +Parentheses are also special characters, so if we want them, we should use `pattern:\(`. The example below looks for a string `"g()"`: + +```js run +alert( "function g()".match(/g\(\)/) ); // "g()" +``` + +If we're looking for a backslash `\`, it's a special character in both regular strings and regexps, so we should double it. + +```js run +alert( "1\\2".match(/\\/) ); // '\' +``` + +## A slash + +A slash symbol `'/'` is not a special character, but in JavaScript it is used to open and close the regexp: `pattern:/...pattern.../`, so we should escape it too. + +Here's what a search for a slash `'/'` looks like: + +```js run +alert( "/".match(/\//) ); // '/' +``` + +On the other hand, if we're not using `pattern:/.../`, but create a regexp using `new RegExp`, then we don't need to escape it: + +```js run +alert( "/".match(new RegExp("/")) ); // finds / +``` + +## new RegExp + +If we are creating a regular expression with `new RegExp`, then we don't have to escape `/`, but need to do some other escaping. + +For instance, consider this: + +```js run +let regexp = new RegExp("\d\.\d"); + +alert( "Chapter 5.1".match(regexp) ); // null +``` + +The similar search in one of previous examples worked with `pattern:/\d\.\d/`, but `new RegExp("\d\.\d")` doesn't work, why? + +The reason is that backslashes are "consumed" by a string. As we may recall, regular strings have their own special characters, such as `\n`, and a backslash is used for escaping. + +Here's how "\d\.\d" is perceived: + +```js run +alert("\d\.\d"); // d.d +``` + +String quotes "consume" backslashes and interpret them on their own, for instance: + +- `\n` -- becomes a newline character, +- `\u1234` -- becomes the Unicode character with such code, +- ...And when there's no special meaning: like `pattern:\d` or `\z`, then the backslash is simply removed. + +So `new RegExp` gets a string without backslashes. That's why the search doesn't work! + +To fix it, we need to double backslashes, because string quotes turn `\\` into `\`: + +```js run +*!* +let regStr = "\\d\\.\\d"; +*/!* +alert(regStr); // \d\.\d (correct now) + +let regexp = new RegExp(regStr); + +alert( "Chapter 5.1".match(regexp) ); // 5.1 +``` + +## Summary + +- To search for special characters `pattern:[ \ ^ $ . | ? * + ( )` literally, we need to prepend them with a backslash `\` ("escape them"). +- We also need to escape `/` if we're inside `pattern:/.../` (but not inside `new RegExp`). +- When passing a string to `new RegExp`, we need to double backslashes `\\`, cause string quotes consume one of them. diff --git a/9-regular-expressions/05-regexp-character-sets-and-ranges/1-find-range-1/solution.md b/9-regular-expressions/08-regexp-character-sets-and-ranges/1-find-range-1/solution.md similarity index 66% rename from 9-regular-expressions/05-regexp-character-sets-and-ranges/1-find-range-1/solution.md rename to 9-regular-expressions/08-regexp-character-sets-and-ranges/1-find-range-1/solution.md index a6d71f661..85c7748f7 100644 --- a/9-regular-expressions/05-regexp-character-sets-and-ranges/1-find-range-1/solution.md +++ b/9-regular-expressions/08-regexp-character-sets-and-ranges/1-find-range-1/solution.md @@ -5,7 +5,7 @@ Answers: **no, yes**. ```js run alert( "Java".match(/Java[^script]/) ); // null ``` -- Yes, because the regexp is case-insensitive, the `pattern:[^script]` part matches the character `"S"`. +- Yes, because the `pattern:[^script]` part matches the character `"S"`. It's not one of `pattern:script`. As the regexp is case-sensitive (no `pattern:i` flag), it treats `"S"` as a different character from `"s"`. ```js run alert( "JavaScript".match(/Java[^script]/) ); // "JavaS" diff --git a/9-regular-expressions/05-regexp-character-sets-and-ranges/1-find-range-1/task.md b/9-regular-expressions/08-regexp-character-sets-and-ranges/1-find-range-1/task.md similarity index 100% rename from 9-regular-expressions/05-regexp-character-sets-and-ranges/1-find-range-1/task.md rename to 9-regular-expressions/08-regexp-character-sets-and-ranges/1-find-range-1/task.md diff --git a/9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md b/9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md similarity index 69% rename from 9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md rename to 9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md index 91568d033..69ade1b19 100644 --- a/9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md +++ b/9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/solution.md @@ -1,8 +1,8 @@ Answer: `pattern:\d\d[-:]\d\d`. ```js run -let reg = /\d\d[-:]\d\d/g; -alert( "Breakfast at 09:00. Dinner at 21-30".match(reg) ); // 09:00, 21-30 +let regexp = /\d\d[-:]\d\d/g; +alert( "Breakfast at 09:00. Dinner at 21-30".match(regexp) ); // 09:00, 21-30 ``` Please note that the dash `pattern:'-'` has a special meaning in square brackets, but only between other characters, not when it's in the beginning or at the end, so we don't need to escape it. diff --git a/9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md b/9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md similarity index 76% rename from 9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md rename to 9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md index 868115bdf..c8441caf4 100644 --- a/9-regular-expressions/05-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md +++ b/9-regular-expressions/08-regexp-character-sets-and-ranges/2-find-time-2-formats/task.md @@ -5,8 +5,8 @@ The time can be in the format `hours:minutes` or `hours-minutes`. Both hours and Write a regexp to find time: ```js -let reg = /your regexp/g; -alert( "Breakfast at 09:00. Dinner at 21-30".match(reg) ); // 09:00, 21-30 +let regexp = /your regexp/g; +alert( "Breakfast at 09:00. Dinner at 21-30".match(regexp) ); // 09:00, 21-30 ``` P.S. In this task we assume that the time is always correct, there's no need to filter out bad strings like "45:67". Later we'll deal with that too. diff --git a/9-regular-expressions/08-regexp-character-sets-and-ranges/article.md b/9-regular-expressions/08-regexp-character-sets-and-ranges/article.md new file mode 100644 index 000000000..b7b82d45f --- /dev/null +++ b/9-regular-expressions/08-regexp-character-sets-and-ranges/article.md @@ -0,0 +1,197 @@ +# Sets and ranges [...] + +Several characters or character classes inside square brackets `[…]` mean to "search for any character among given". + +## Sets + +For instance, `pattern:[eao]` means any of the 3 characters: `'a'`, `'e'`, or `'o'`. + +That's called a *set*. Sets can be used in a regexp along with regular characters: + +```js run +// find [t or m], and then "op" +alert( "Mop top".match(/[tm]op/gi) ); // "Mop", "top" +``` + +Please note that although there are multiple characters in the set, they correspond to exactly one character in the match. + +So the example below gives no matches: + +```js run +// find "V", then [o or i], then "la" +alert( "Voila".match(/V[oi]la/) ); // null, no matches +``` + +The pattern searches for: + +- `pattern:V`, +- then *one* of the letters `pattern:[oi]`, +- then `pattern:la`. + +So there would be a match for `match:Vola` or `match:Vila`. + +## Ranges + +Square brackets may also contain *character ranges*. + +For instance, `pattern:[a-z]` is a character in range from `a` to `z`, and `pattern:[0-5]` is a digit from `0` to `5`. + +In the example below we're searching for `"x"` followed by two digits or letters from `A` to `F`: + +```js run +alert( "Exception 0xAF".match(/x[0-9A-F][0-9A-F]/g) ); // xAF +``` + +Here `pattern:[0-9A-F]` has two ranges: it searches for a character that is either a digit from `0` to `9` or a letter from `A` to `F`. + +If we'd like to look for lowercase letters as well, we can add the range `a-f`: `pattern:[0-9A-Fa-f]`. Or add the flag `pattern:i`. + +We can also use character classes inside `[…]`. + +For instance, if we'd like to look for a wordly character `pattern:\w` or a hyphen `pattern:-`, then the set is `pattern:[\w-]`. + +Combining multiple classes is also possible, e.g. `pattern:[\s\d]` means "a space character or a digit". + +```smart header="Character classes are shorthands for certain character sets" +For instance: + +- **\d** -- is the same as `pattern:[0-9]`, +- **\w** -- is the same as `pattern:[a-zA-Z0-9_]`, +- **\s** -- is the same as `pattern:[\t\n\v\f\r ]`, plus few other rare Unicode space characters. +``` + +### Example: multi-language \w + +As the character class `pattern:\w` is a shorthand for `pattern:[a-zA-Z0-9_]`, it can't find Chinese hieroglyphs, Cyrillic letters, etc. + +We can write a more universal pattern, that looks for wordly characters in any language. That's easy with Unicode properties: `pattern:[\p{Alpha}\p{M}\p{Nd}\p{Pc}\p{Join_C}]`. + +Let's decipher it. Similar to `pattern:\w`, we're making a set of our own that includes characters with following Unicode properties: + +- `Alphabetic` (`Alpha`) - for letters, +- `Mark` (`M`) - for accents, +- `Decimal_Number` (`Nd`) - for digits, +- `Connector_Punctuation` (`Pc`) - for the underscore `'_'` and similar characters, +- `Join_Control` (`Join_C`) - two special codes `200c` and `200d`, used in ligatures, e.g. in Arabic. + +An example of use: + +```js run +let regexp = /[\p{Alpha}\p{M}\p{Nd}\p{Pc}\p{Join_C}]/gu; + +let str = `Hi 你好 12`; + +// finds all letters and digits: +alert( str.match(regexp) ); // H,i,你,好,1,2 +``` + +Of course, we can edit this pattern: add Unicode properties or remove them. Unicode properties are covered in more details in the article <info:regexp-unicode>. + +```warn header="Unicode properties aren't supported in IE" +Unicode properties `pattern:p{…}` are not implemented in IE. If we really need them, we can use library [XRegExp](https://xregexp.com/). + +Or just use ranges of characters in a language that interests us, e.g. `pattern:[а-я]` for Cyrillic letters. +``` + +## Excluding ranges + +Besides normal ranges, there are "excluding" ranges that look like `pattern:[^…]`. + +They are denoted by a caret character `^` at the start and match any character *except the given ones*. + +For instance: + +- `pattern:[^aeyo]` -- any character except `'a'`, `'e'`, `'y'` or `'o'`. +- `pattern:[^0-9]` -- any character except a digit, the same as `pattern:\D`. +- `pattern:[^\s]` -- any non-space character, same as `\S`. + +The example below looks for any characters except letters, digits and spaces: + +```js run +alert( "alice15@gmail.com".match(/[^\d\sA-Z]/gi) ); // @ and . +``` + +## Escaping in […] + +Usually when we want to find exactly a special character, we need to escape it like `pattern:\.`. And if we need a backslash, then we use `pattern:\\`, and so on. + +In square brackets we can use the vast majority of special characters without escaping: + +- Symbols `pattern:. + ( )` never need escaping. +- A hyphen `pattern:-` is not escaped in the beginning or the end (where it does not define a range). +- A caret `pattern:^` is only escaped in the beginning (where it means exclusion). +- The closing square bracket `pattern:]` is always escaped (if we need to look for that symbol). + +In other words, all special characters are allowed without escaping, except when they mean something for square brackets. + +A dot `.` inside square brackets means just a dot. The pattern `pattern:[.,]` would look for one of characters: either a dot or a comma. + +In the example below the regexp `pattern:[-().^+]` looks for one of the characters `-().^+`: + +```js run +// No need to escape +let regexp = /[-().^+]/g; + +alert( "1 + 2 - 3".match(regexp) ); // Matches +, - +``` + +...But if you decide to escape them "just in case", then there would be no harm: + +```js run +// Escaped everything +let regexp = /[\-\(\)\.\^\+]/g; + +alert( "1 + 2 - 3".match(regexp) ); // also works: +, - +``` + +## Ranges and flag "u" + +If there are surrogate pairs in the set, flag `pattern:u` is required for them to work correctly. + +For instance, let's look for `pattern:[𝒳𝒴]` in the string `subject:𝒳`: + +```js run +alert( '𝒳'.match(/[𝒳𝒴]/) ); // shows a strange character, like [?] +// (the search was performed incorrectly, half-character returned) +``` + +The result is incorrect, because by default regular expressions "don't know" about surrogate pairs. + +The regular expression engine thinks that `[𝒳𝒴]` -- are not two, but four characters: +1. left half of `𝒳` `(1)`, +2. right half of `𝒳` `(2)`, +3. left half of `𝒴` `(3)`, +4. right half of `𝒴` `(4)`. + +We can see their codes like this: + +```js run +for(let i=0; i<'𝒳𝒴'.length; i++) { + alert('𝒳𝒴'.charCodeAt(i)); // 55349, 56499, 55349, 56500 +}; +``` + +So, the example above finds and shows the left half of `𝒳`. + +If we add flag `pattern:u`, then the behavior will be correct: + +```js run +alert( '𝒳'.match(/[𝒳𝒴]/u) ); // 𝒳 +``` + +The similar situation occurs when looking for a range, such as `[𝒳-𝒴]`. + +If we forget to add flag `pattern:u`, there will be an error: + +```js run +'𝒳'.match(/[𝒳-𝒴]/); // Error: Invalid regular expression +``` + +The reason is that without flag `pattern:u` surrogate pairs are perceived as two characters, so `[𝒳-𝒴]` is interpreted as `[<55349><56499>-<55349><56500>]` (every surrogate pair is replaced with its codes). Now it's easy to see that the range `56499-55349` is invalid: its starting code `56499` is greater than the end `55349`. That's the formal reason for the error. + +With the flag `pattern:u` the pattern works correctly: + +```js run +// look for characters from 𝒳 to 𝒵 +alert( '𝒴'.match(/[𝒳-𝒵]/u) ); // 𝒴 +``` diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/solution.md b/9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/solution.md deleted file mode 100644 index c066f3e36..000000000 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/solution.md +++ /dev/null @@ -1,17 +0,0 @@ -We need to find the beginning of the comment `match:<!--`, then everything till the end of `match:-->`. - -The first idea could be `pattern:<!--.*?-->` -- the lazy quantifier makes the dot stop right before `match:-->`. - -But a dot in JavaScript means "any symbol except the newline". So multiline comments won't be found. - -We can use `pattern:[\s\S]` instead of the dot to match "anything": - -```js run -let reg = /<!--[\s\S]*?-->/g; - -let str = `... <!-- My -- comment - test --> .. <!----> .. -`; - -alert( str.match(reg) ); // '<!-- My -- comment \n test -->', '<!---->' -``` diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md b/9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md deleted file mode 100644 index 2cf48a4ae..000000000 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md +++ /dev/null @@ -1,15 +0,0 @@ -# Find HTML tags - -Create a regular expression to find all (opening and closing) HTML tags with their attributes. - -An example of use: - -```js run -let reg = /your regexp/g; - -let str = '<> <a href="/"> <input type="radio" checked> <b>'; - -alert( str.match(reg) ); // '<a href="/">', '<input type="radio" checked>', '<b>' -``` - -Let's assume that may not contain `<` and `>` inside (in quotes too), that simplifies things a bit. diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy1.svg b/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy1.svg deleted file mode 100644 index 65e490e97..000000000 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy1.svg +++ /dev/null @@ -1,16 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> -<svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> - <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> - <title>witch_greedy1.svg</title> - <desc>Created with sketchtool.</desc> - <g id="regexp" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> - <g id="witch_greedy1.svg"> - <text id="a-"witch"-and-her-"b" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal" fill="#8A704D"> - <tspan x="20" y="112">a "witch" and her "broom" is one</tspan> - </text> - <rect id="Rectangle-1" stroke="#E8C48E" x="45.5" y="50.5" width="14" height="76"></rect> - <path d="M52.0039062,60.4414062 L51.6757812,65.9375 L49.7539062,65.9375 L49.4140625,60.4414062 L52.0039062,60.4414062 Z M56.5625,60.4414062 L56.234375,65.9375 L54.3125,65.9375 L53.9726562,60.4414062 L56.5625,60.4414062 Z" id="--"-----------------" fill="#CB1E31"></path> - <path id="Line" d="M51.5,30.5 L51.5,8.5 L53.5,8.5 L53.5,30.5 L59.5,30.5 L52.5,44.5 L45.5,30.5 L51.5,30.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> - </g> - </g> -</svg> \ No newline at end of file diff --git a/9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/solution.md b/9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/solution.md deleted file mode 100644 index d653ff970..000000000 --- a/9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/solution.md +++ /dev/null @@ -1,29 +0,0 @@ -A regexp to search 3-digit color `#abc`: `pattern:/#[a-f0-9]{3}/i`. - -We can add exactly 3 more optional hex digits. We don't need more or less. Either we have them or we don't. - -The simplest way to add them -- is to append to the regexp: `pattern:/#[a-f0-9]{3}([a-f0-9]{3})?/i` - -We can do it in a smarter way though: `pattern:/#([a-f0-9]{3}){1,2}/i`. - -Here the regexp `pattern:[a-f0-9]{3}` is in parentheses to apply the quantifier `pattern:{1,2}` to it as a whole. - -In action: - -```js run -let reg = /#([a-f0-9]{3}){1,2}/gi; - -let str = "color: #3f3; background-color: #AA00ef; and: #abcd"; - -alert( str.match(reg) ); // #3f3 #AA00ef #abc -``` - -There's a minor problem here: the pattern found `match:#abc` in `subject:#abcd`. To prevent that we can add `pattern:\b` to the end: - -```js run -let reg = /#([a-f0-9]{3}){1,2}\b/gi; - -let str = "color: #3f3; background-color: #AA00ef; and: #abcd"; - -alert( str.match(reg) ); // #3f3 #AA00ef -``` diff --git a/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/solution.md b/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/solution.md deleted file mode 100644 index 23065413e..000000000 --- a/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/solution.md +++ /dev/null @@ -1,18 +0,0 @@ - -An non-negative integer number is `pattern:\d+`. We should exclude `0` as the first digit, as we don't need zero, but we can allow it in further digits. - -So that gives us `pattern:[1-9]\d*`. - -A decimal part is: `pattern:\.\d+`. - -Because the decimal part is optional, let's put it in parentheses with the quantifier `pattern:'?'`. - -Finally we have the regexp: `pattern:[1-9]\d*(\.\d+)?`: - -```js run -let reg = /[1-9]\d*(\.\d+)?/g; - -let str = "1.5 0 -5 12. 123.4."; - -alert( str.match(reg) ); // 1.5, 0, 12, 123.4 -``` diff --git a/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/task.md b/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/task.md deleted file mode 100644 index ad8c81eae..000000000 --- a/9-regular-expressions/09-regexp-groups/3-find-decimal-positive-numbers/task.md +++ /dev/null @@ -1,12 +0,0 @@ -# Find positive numbers - -Create a regexp that looks for positive numbers, including those without a decimal point. - -An example of use: -```js -let reg = /your regexp/g; - -let str = "1.5 0 -5 12. 123.4."; - -alert( str.match(reg) ); // 1.5, 12, 123.4 (ignores 0 and -5) -``` diff --git a/9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/solution.md b/9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/solution.md deleted file mode 100644 index dd2410847..000000000 --- a/9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/solution.md +++ /dev/null @@ -1,11 +0,0 @@ -A positive number with an optional decimal part is (per previous task): `pattern:\d+(\.\d+)?`. - -Let's add an optional `-` in the beginning: - -```js run -let reg = /-?\d+(\.\d+)?/g; - -let str = "-1.5 0 2 -123.4."; - -alert( str.match(reg) ); // -1.5, 0, 2, -123.4 -``` diff --git a/9-regular-expressions/09-regexp-groups/5-parse-expression/solution.md b/9-regular-expressions/09-regexp-groups/5-parse-expression/solution.md deleted file mode 100644 index 3db5f667c..000000000 --- a/9-regular-expressions/09-regexp-groups/5-parse-expression/solution.md +++ /dev/null @@ -1,51 +0,0 @@ -A regexp for a number is: `pattern:-?\d+(\.\d+)?`. We created it in previous tasks. - -An operator is `pattern:[-+*/]`. We put the dash `pattern:-` first, because in the middle it would mean a character range, we don't need that. - -Note that a slash should be escaped inside a JavaScript regexp `pattern:/.../`. - -We need a number, an operator, and then another number. And optional spaces between them. - -The full regular expression: `pattern:-?\d+(\.\d+)?\s*[-+*/]\s*-?\d+(\.\d+)?`. - -To get a result as an array let's put parentheses around the data that we need: numbers and the operator: `pattern:(-?\d+(\.\d+)?)\s*([-+*/])\s*(-?\d+(\.\d+)?)`. - -In action: - -```js run -let reg = /(-?\d+(\.\d+)?)\s*([-+*\/])\s*(-?\d+(\.\d+)?)/; - -alert( "1.2 + 12".match(reg) ); -``` - -The result includes: - -- `result[0] == "1.2 + 12"` (full match) -- `result[1] == "1.2"` (first group `(-?\d+(\.\d+)?)` -- the first number, including the decimal part) -- `result[2] == ".2"` (second group`(\.\d+)?` -- the first decimal part) -- `result[3] == "+"` (third group `([-+*\/])` -- the operator) -- `result[4] == "12"` (forth group `(-?\d+(\.\d+)?)` -- the second number) -- `result[5] == undefined` (fifth group `(\.\d+)?` -- the last decimal part is absent, so it's undefined) - -We only want the numbers and the operator, without the full match or the decimal parts. - -The full match (the arrays first item) can be removed by shifting the array `pattern:result.shift()`. - -The decimal groups can be removed by making them into non-capturing groups, by adding `pattern:?:` to the beginning: `pattern:(?:\.\d+)?`. - -The final solution: - -```js run -function parse(expr) { - let reg = /(-?\d+(?:\.\d+)?)\s*([-+*\/])\s*(-?\d+(?:\.\d+)?)/; - - let result = expr.match(reg); - - if (!result) return []; - result.shift(); - - return result; -} - -alert( parse("-1.23 * 3.45") ); // -1.23, *, 3.45 -``` diff --git a/9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/solution.md b/9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/solution.md similarity index 57% rename from 9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/solution.md rename to 9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/solution.md index d4ddb1369..21b8762ec 100644 --- a/9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/solution.md +++ b/9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/solution.md @@ -2,8 +2,8 @@ Solution: ```js run -let reg = /\.{3,}/g; -alert( "Hello!... How goes?.....".match(reg) ); // ..., ..... +let regexp = /\.{3,}/g; +alert( "Hello!... How goes?.....".match(regexp) ); // ..., ..... ``` Please note that the dot is a special character, so we have to escape it and insert as `\.`. diff --git a/9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/task.md b/9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/task.md similarity index 59% rename from 9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/task.md rename to 9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/task.md index 6fd91bdcf..4140b4a98 100644 --- a/9-regular-expressions/07-regexp-quantifiers/1-find-text-manydots/task.md +++ b/9-regular-expressions/09-regexp-quantifiers/1-find-text-manydots/task.md @@ -9,6 +9,6 @@ Create a regexp to find ellipsis: 3 (or more?) dots in a row. Check it: ```js -let reg = /your regexp/g; -alert( "Hello!... How goes?.....".match(reg) ); // ..., ..... +let regexp = /your regexp/g; +alert( "Hello!... How goes?.....".match(regexp) ); // ..., ..... ``` diff --git a/9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/solution.md b/9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/solution.md similarity index 60% rename from 9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/solution.md rename to 9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/solution.md index ec871d05c..afee89c50 100644 --- a/9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/solution.md +++ b/9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/solution.md @@ -1,23 +1,23 @@ -We need to look for `#` followed by 6 hexadimal characters. +We need to look for `#` followed by 6 hexadecimal characters. -A hexadimal character can be described as `pattern:[0-9a-fA-F]`. Or if we use the `i` flag, then just `pattern:[0-9a-f]`. +A hexadecimal character can be described as `pattern:[0-9a-fA-F]`. Or if we use the `pattern:i` flag, then just `pattern:[0-9a-f]`. Then we can look for 6 of them using the quantifier `pattern:{6}`. As a result, we have the regexp: `pattern:/#[a-f0-9]{6}/gi`. ```js run -let reg = /#[a-f0-9]{6}/gi; +let regexp = /#[a-f0-9]{6}/gi; let str = "color:#121212; background-color:#AA00ef bad-colors:f#fddee #fd2" -alert( str.match(reg) ); // #121212,#AA00ef +alert( str.match(regexp) ); // #121212,#AA00ef ``` The problem is that it finds the color in longer sequences: ```js run -alert( "#12345678".match( /#[a-f0-9]{6}/gi ) ) // #12345678 +alert( "#12345678".match( /#[a-f0-9]{6}/gi ) ) // #123456 ``` To fix that, we can add `pattern:\b` to the end: diff --git a/9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/task.md b/9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/task.md similarity index 71% rename from 9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/task.md rename to 9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/task.md index 1960a09c6..9a1923a7e 100644 --- a/9-regular-expressions/07-regexp-quantifiers/2-find-html-colors-6hex/task.md +++ b/9-regular-expressions/09-regexp-quantifiers/2-find-html-colors-6hex/task.md @@ -1,15 +1,15 @@ # Regexp for HTML colors -Create a regexp to search HTML-colors written as `#ABCDEF`: first `#` and then 6 hexadimal characters. +Create a regexp to search HTML-colors written as `#ABCDEF`: first `#` and then 6 hexadecimal characters. An example of use: ```js -let reg = /...your regexp.../ +let regexp = /...your regexp.../ let str = "color:#121212; background-color:#AA00ef bad-colors:f#fddee #fd2 #12345678"; -alert( str.match(reg) ) // #121212,#AA00ef +alert( str.match(regexp) ) // #121212,#AA00ef ``` P.S. In this task we do not need other color formats like `#123` or `rgb(1,2,3)` etc. diff --git a/9-regular-expressions/07-regexp-quantifiers/article.md b/9-regular-expressions/09-regexp-quantifiers/article.md similarity index 63% rename from 9-regular-expressions/07-regexp-quantifiers/article.md rename to 9-regular-expressions/09-regexp-quantifiers/article.md index 5ab592561..1a7eecfeb 100644 --- a/9-regular-expressions/07-regexp-quantifiers/article.md +++ b/9-regular-expressions/09-regexp-quantifiers/article.md @@ -2,7 +2,7 @@ Let's say we have a string like `+7(903)-123-45-67` and want to find all numbers in it. But unlike before, we are interested not in single digits, but full numbers: `7, 903, 123, 45, 67`. -A number is a sequence of 1 or more digits `\d`. To mark how many we need, we need to append a *quantifier*. +A number is a sequence of 1 or more digits `pattern:\d`. To mark how many we need, we can append a *quantifier*. ## Quantity {n} @@ -12,7 +12,7 @@ A quantifier is appended to a character (or a character class, or a `[...]` set It has a few advanced forms, let's see examples: -The exact count: `{5}` +The exact count: `pattern:{5}` : `pattern:\d{5}` denotes exactly 5 digits, the same as `pattern:\d\d\d\d\d`. The example below looks for a 5-digit number: @@ -23,7 +23,7 @@ The exact count: `{5}` We can add `\b` to exclude longer numbers: `pattern:\b\d{5}\b`. -The range: `{3,5}`, match 3-5 times +The range: `pattern:{3,5}`, match 3-5 times : To find numbers from 3 to 5 digits we can put the limits into curly braces: `pattern:\d{3,5}` ```js run @@ -54,8 +54,8 @@ alert(numbers); // 7,903,123,45,67 There are shorthands for most used quantifiers: -`+` -: Means "one or more", the same as `{1,}`. +`pattern:+` +: Means "one or more", the same as `pattern:{1,}`. For instance, `pattern:\d+` looks for numbers: @@ -65,8 +65,8 @@ There are shorthands for most used quantifiers: alert( str.match(/\d+/g) ); // 7,903,123,45,67 ``` -`?` -: Means "zero or one", the same as `{0,1}`. In other words, it makes the symbol optional. +`pattern:?` +: Means "zero or one", the same as `pattern:{0,1}`. In other words, it makes the symbol optional. For instance, the pattern `pattern:ou?r` looks for `match:o` followed by zero or one `match:u`, and then `match:r`. @@ -78,16 +78,16 @@ There are shorthands for most used quantifiers: alert( str.match(/colou?r/g) ); // color, colour ``` -`*` -: Means "zero or more", the same as `{0,}`. That is, the character may repeat any times or be absent. +`pattern:*` +: Means "zero or more", the same as `pattern:{0,}`. That is, the character may repeat any times or be absent. - For example, `pattern:\d0*` looks for a digit followed by any number of zeroes: + For example, `pattern:\d0*` looks for a digit followed by any number of zeroes (may be many or none): ```js run alert( "100 10 1".match(/\d0*/g) ); // 100, 10, 1 ``` - Compare it with `'+'` (one or more): + Compare it with `pattern:+` (one or more): ```js run alert( "100 10 1".match(/\d0+/g) ); // 100, 10 @@ -98,43 +98,45 @@ There are shorthands for most used quantifiers: Quantifiers are used very often. They serve as the main "building block" of complex regular expressions, so let's see more examples. -Regexp "decimal fraction" (a number with a floating point): `pattern:\d+\.\d+` -: In action: - ```js run - alert( "0 1 12.345 7890".match(/\d+\.\d+/g) ); // 12.345 - ``` +**Regexp for decimal fractions (a number with a floating point): `pattern:\d+\.\d+`** -Regexp "open HTML-tag without attributes", like `<span>` or `<p>`: `pattern:/<[a-z]+>/i` -: In action: +In action: +```js run +alert( "0 1 12.345 7890".match(/\d+\.\d+/g) ); // 12.345 +``` + +**Regexp for an "opening HTML-tag without attributes", such as `<span>` or `<p>`.** + +1. The simplest one: `pattern:/<[a-z]+>/i` ```js run alert( "<body> ... </body>".match(/<[a-z]+>/gi) ); // <body> ``` - We look for character `pattern:'<'` followed by one or more English letters, and then `pattern:'>'`. + The regexp looks for character `pattern:'<'` followed by one or more Latin letters, and then `pattern:'>'`. + +2. Improved: `pattern:/<[a-z][a-z0-9]*>/i` -Regexp "open HTML-tag without attributes" (improved): `pattern:/<[a-z][a-z0-9]*>/i` -: Better regexp: according to the standard, HTML tag name may have a digit at any position except the first one, like `<h1>`. + According to the standard, HTML tag name may have a digit at any position except the first one, like `<h1>`. ```js run alert( "<h1>Hi!</h1>".match(/<[a-z][a-z0-9]*>/gi) ); // <h1> ``` -Regexp "opening or closing HTML-tag without attributes": `pattern:/<\/?[a-z][a-z0-9]*>/i` -: We added an optional slash `pattern:/?` before the tag. Had to escape it with a backslash, otherwise JavaScript would think it is the pattern end. +**Regexp "opening or closing HTML-tag without attributes": `pattern:/<\/?[a-z][a-z0-9]*>/i`** - ```js run - alert( "<h1>Hi!</h1>".match(/<\/?[a-z][a-z0-9]*>/gi) ); // <h1>, </h1> - ``` +We added an optional slash `pattern:/?` near the beginning of the pattern. Had to escape it with a backslash, otherwise JavaScript would think it is the pattern end. + +```js run +alert( "<h1>Hi!</h1>".match(/<\/?[a-z][a-z0-9]*>/gi) ); // <h1>, </h1> +``` ```smart header="To make a regexp more precise, we often need make it more complex" We can see one common rule in these examples: the more precise is the regular expression -- the longer and more complex it is. -For instance, for HTML tags we could use a simpler regexp: `pattern:<\w+>`. - -...But because `pattern:\w` means any English letter or a digit or `'_'`, the regexp also matches non-tags, for instance `match:<_>`. So it's much simpler than `pattern:<[a-z][a-z0-9]*>`, but less reliable. +For instance, for HTML tags we could use a simpler regexp: `pattern:<\w+>`. But as HTML has stricter restrictions for a tag name, `pattern:<[a-z][a-z0-9]*>` is more reliable. -Are we ok with `pattern:<\w+>` or we need `pattern:<[a-z][a-z0-9]*>`? +Can we use `pattern:<\w+>` or we need `pattern:<[a-z][a-z0-9]*>`? -In real life both variants are acceptable. Depends on how tolerant we can be to "extra" matches and whether it's difficult or not to filter them out by other means. +In real life both variants are acceptable. Depends on how tolerant we can be to "extra" matches and whether it's difficult or not to remove them from the result by other means. ``` diff --git a/9-regular-expressions/10-regexp-backreferences/article.md b/9-regular-expressions/10-regexp-backreferences/article.md deleted file mode 100644 index a7a934e4a..000000000 --- a/9-regular-expressions/10-regexp-backreferences/article.md +++ /dev/null @@ -1,65 +0,0 @@ -# Backreferences in pattern: \n and \k - -Capturing groups can be accessed not only in the result or in the replacement string, but also in the pattern itself. - -## Backreference by number: \n - -A group can be referenced in the pattern using `\n`, where `n` is the group number. - -To make things clear let's consider a task. - -We need to find a quoted string: either a single-quoted `subject:'...'` or a double-quoted `subject:"..."` -- both variants need to match. - -How to look for them? - -We can put two kinds of quotes in the pattern: `pattern:['"](.*?)['"]`, but it would find strings with mixed quotes, like `match:"...'` and `match:'..."`. That would lead to incorrect matches when one quote appears inside other ones, like the string `subject:"She's the one!"`: - -```js run -let str = `He said: "She's the one!".`; - -let reg = /['"](.*?)['"]/g; - -// The result is not what we expect -alert( str.match(reg) ); // "She' -``` - -As we can see, the pattern found an opening quote `match:"`, then the text is consumed lazily till the other quote `match:'`, that closes the match. - -To make sure that the pattern looks for the closing quote exactly the same as the opening one, we can make a groups of it and use the backreference. - -Here's the correct code: - -```js run -let str = `He said: "She's the one!".`; - -*!* -let reg = /(['"])(.*?)\1/g; -*/!* - -alert( str.match(reg) ); // "She's the one!" -``` - -Now it works! The regular expression engine finds the first quote `pattern:(['"])` and remembers the content of `pattern:(...)`, that's the first capturing group. - -Further in the pattern `pattern:\1` means "find the same text as in the first group", exactly the same quote in our case. - -Please note: - -- To reference a group inside a replacement string -- we use `$1`, while in the pattern -- a backslash `\1`. -- If we use `?:` in the group, then we can't reference it. Groups that are excluded from capturing `(?:...)` are not remembered by the engine. - -## Backreference by name: `\k<name>` - -For named groups, we can backreference by `\k<name>`. - -The same example with the named group: - -```js run -let str = `He said: "She's the one!".`; - -*!* -let reg = /(?<quote>['"])(.*?)\k<quote>/g; -*/!* - -alert( str.match(reg) ); // "She's the one!" -``` diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/1-lazy-greedy/solution.md b/9-regular-expressions/10-regexp-greedy-and-lazy/1-lazy-greedy/solution.md similarity index 100% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/1-lazy-greedy/solution.md rename to 9-regular-expressions/10-regexp-greedy-and-lazy/1-lazy-greedy/solution.md diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/1-lazy-greedy/task.md b/9-regular-expressions/10-regexp-greedy-and-lazy/1-lazy-greedy/task.md similarity index 56% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/1-lazy-greedy/task.md rename to 9-regular-expressions/10-regexp-greedy-and-lazy/1-lazy-greedy/task.md index b46f55917..596f61a4e 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/1-lazy-greedy/task.md +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/1-lazy-greedy/task.md @@ -3,5 +3,5 @@ What's the match here? ```js -"123 456".match(/\d+? \d+?/g) ); // ? +alert( "123 456".match(/\d+? \d+?/g) ); // ? ``` diff --git a/9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/solution.md b/9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/solution.md new file mode 100644 index 000000000..0244963d1 --- /dev/null +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/solution.md @@ -0,0 +1,15 @@ +We need to find the beginning of the comment `match:<!--`, then everything till the end of `match:-->`. + +An acceptable variant is `pattern:<!--.*?-->` -- the lazy quantifier makes the dot stop right before `match:-->`. We also need to add flag `pattern:s` for the dot to include newlines. + +Otherwise multiline comments won't be found: + +```js run +let regexp = /<!--.*?-->/gs; + +let str = `... <!-- My -- comment + test --> .. <!----> .. +`; + +alert( str.match(regexp) ); // '<!-- My -- comment \n test -->', '<!---->' +``` diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/task.md b/9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/task.md similarity index 56% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/task.md rename to 9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/task.md index 81fd5c634..551d9c725 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/3-find-html-comments/task.md +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/3-find-html-comments/task.md @@ -3,11 +3,11 @@ Find all HTML comments in the text: ```js -let reg = /your regexp/g; +let regexp = /your regexp/g; let str = `... <!-- My -- comment test --> .. <!----> .. `; -alert( str.match(reg) ); // '<!-- My -- comment \n test -->', '<!---->' +alert( str.match(regexp) ); // '<!-- My -- comment \n test -->', '<!---->' ``` diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md b/9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md similarity index 51% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md rename to 9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md index c453926fa..b4d9f7496 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/solution.md @@ -2,9 +2,9 @@ The solution is `pattern:<[^<>]+>`. ```js run -let reg = /<[^<>]+>/g; +let regexp = /<[^<>]+>/g; let str = '<> <a href="/"> <input type="radio" checked> <b>'; -alert( str.match(reg) ); // '<a href="/">', '<input type="radio" checked>', '<b>' +alert( str.match(regexp) ); // '<a href="/">', '<input type="radio" checked>', '<b>' ``` diff --git a/9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md b/9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md new file mode 100644 index 000000000..6759152ff --- /dev/null +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/4-find-html-tags-greedy-lazy/task.md @@ -0,0 +1,15 @@ +# Find HTML tags + +Create a regular expression to find all (opening and closing) HTML tags with their attributes. + +An example of use: + +```js run +let regexp = /your regexp/g; + +let str = '<> <a href="/"> <input type="radio" checked> <b>'; + +alert( str.match(regexp) ); // '<a href="/">', '<input type="radio" checked>', '<b>' +``` + +Here we assume that tag attributes may not contain `<` and `>` (inside quotes too), that simplifies things a bit. diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/article.md b/9-regular-expressions/10-regexp-greedy-and-lazy/article.md similarity index 70% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/article.md rename to 9-regular-expressions/10-regexp-greedy-and-lazy/article.md index f3e9219cb..e20175075 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/article.md +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/article.md @@ -8,7 +8,7 @@ Let's take the following task as an example. We have a text and need to replace all quotes `"..."` with guillemet marks: `«...»`. They are preferred for typography in many countries. -For instance: `"Hello, world"` should become `«Hello, world»`. Some countries prefer other quotes, like `„Witam, świat!”` (Polish) or `「你好,世界」` (Chinese), but for our task let's choose `«...»`. +For instance: `"Hello, world"` should become `«Hello, world»`. There exist other quotes, such as `„Witaj, świecie!”` (Polish) or `「你好,世界」` (Chinese), but for our task let's choose `«...»`. The first thing to do is to locate quoted strings, and then we can replace them. @@ -17,11 +17,11 @@ A regular expression like `pattern:/".+"/g` (a quote, then something, then the o Let's try it: ```js run -let reg = /".+"/g; +let regexp = /".+"/g; let str = 'a "witch" and her "broom" is one'; -alert( str.match(reg) ); // "witch" and her "broom" +alert( str.match(regexp) ); // "witch" and her "broom" ``` ...We can see that it works not as intended! @@ -35,7 +35,7 @@ That can be described as "greediness is the cause of all evil". To find a match, the regular expression engine uses the following algorithm: - For every position in the string - - Match the pattern at that position. + - Try to match the pattern at that position. - If there's no match, go to the next position. These common words do not make it obvious why the regexp fails, so let's elaborate how the search works for the pattern `pattern:".+"`. @@ -44,7 +44,7 @@ These common words do not make it obvious why the regexp fails, so let's elabora The regular expression engine tries to find it at the zero position of the source string `subject:a "witch" and her "broom" is one`, but there's `subject:a` there, so there's immediately no match. - Then it advances: goes to the next positions in the source string and tries to find the first character of the pattern there, and finally finds the quote at the 3rd position: + Then it advances: goes to the next positions in the source string and tries to find the first character of the pattern there, fails again, and finally finds the quote at the 3rd position:  @@ -54,13 +54,13 @@ These common words do not make it obvious why the regexp fails, so let's elabora  -3. Then the dot repeats because of the quantifier `pattern:.+`. The regular expression engine builds the match by taking characters one by one while it is possible. +3. Then the dot repeats because of the quantifier `pattern:.+`. The regular expression engine adds to the match one character after another. - ...When it becomes impossible? All characters match the dot, so it only stops when it reaches the end of the string: + ...Until when? All characters match the dot, so it only stops when it reaches the end of the string:  -4. Now the engine finished repeating for `pattern:.+` and tries to find the next character of the pattern. It's the quote `pattern:"`. But there's a problem: the string has finished, there are no more characters! +4. Now the engine finished repeating `pattern:.+` and tries to find the next character of the pattern. It's the quote `pattern:"`. But there's a problem: the string has finished, there are no more characters! The regular expression engine understands that it took too many `pattern:.+` and starts to *backtrack*. @@ -68,9 +68,9 @@ These common words do not make it obvious why the regexp fails, so let's elabora  - Now it assumes that `pattern:.+` ends one character before the end and tries to match the rest of the pattern from that position. + Now it assumes that `pattern:.+` ends one character before the string end and tries to match the rest of the pattern from that position. - If there were a quote there, then that would be the end, but the last character is `subject:'e'`, so there's no match. + If there were a quote there, then the search would end, but the last character is `subject:'e'`, so there's no match. 5. ...So the engine decreases the number of repetitions of `pattern:.+` by one more character: @@ -84,19 +84,19 @@ These common words do not make it obvious why the regexp fails, so let's elabora 7. The match is complete. -8. So the first match is `match:"witch" and her "broom"`. The further search starts where the first match ends, but there are no more quotes in the rest of the string `subject:is one`, so no more results. +8. So the first match is `match:"witch" and her "broom"`. If the regular expression has flag `pattern:g`, then the search will continue from where the first match ends. There are no more quotes in the rest of the string `subject:is one`, so no more results. That's probably not what we expected, but that's how it works. -**In the greedy mode (by default) the quantifier is repeated as many times as possible.** +**In the greedy mode (by default) a quantified character is repeated as many times as possible.** -The regexp engine tries to fetch as many characters as it can by `pattern:.+`, and then shortens that one by one. +The regexp engine adds to the match as many characters as it can for `pattern:.+`, and then shortens that one by one, if the rest of the pattern doesn't match. -For our task we want another thing. That's what the lazy quantifier mode is for. +For our task we want another thing. That's where a lazy mode can help. ## Lazy mode -The lazy mode of quantifier is an opposite to the greedy mode. It means: "repeat minimal number of times". +The lazy mode of quantifiers is an opposite to the greedy mode. It means: "repeat minimal number of times". We can enable it by putting a question mark `pattern:'?'` after the quantifier, so that it becomes `pattern:*?` or `pattern:+?` or even `pattern:??` for `pattern:'?'`. @@ -105,11 +105,11 @@ To make things clear: usually a question mark `pattern:?` is a quantifier by its The regexp `pattern:/".+?"/g` works as intended: it finds `match:"witch"` and `match:"broom"`: ```js run -let reg = /".+?"/g; +let regexp = /".+?"/g; let str = 'a "witch" and her "broom" is one'; -alert( str.match(reg) ); // witch, broom +alert( str.match(regexp) ); // "witch", "broom" ``` To clearly understand the change, let's trace the search step by step. @@ -140,7 +140,7 @@ To clearly understand the change, let's trace the search step by step.  -In this example we saw how the lazy mode works for `pattern:+?`. Quantifiers `pattern:+?` and `pattern:??` work the similar way -- the regexp engine increases the number of repetitions only if the rest of the pattern can't match on the given position. +In this example we saw how the lazy mode works for `pattern:+?`. Quantifiers `pattern:*?` and `pattern:??` work the similar way -- the regexp engine increases the number of repetitions only if the rest of the pattern can't match on the given position. **Laziness is only enabled for the quantifier with `?`.** @@ -149,20 +149,19 @@ Other quantifiers remain greedy. For instance: ```js run -alert( "123 456".match(/\d+ \d+?/g) ); // 123 4 +alert( "123 456".match(/\d+ \d+?/) ); // 123 4 ``` -1. The pattern `pattern:\d+` tries to match as many numbers as it can (greedy mode), so it finds `match:123` and stops, because the next character is a space `pattern:' '`. -2. Then there's a space in pattern, it matches. +1. The pattern `pattern:\d+` tries to match as many digits as it can (greedy mode), so it finds `match:123` and stops, because the next character is a space `pattern:' '`. +2. Then there's a space in the pattern, it matches. 3. Then there's `pattern:\d+?`. The quantifier is in lazy mode, so it finds one digit `match:4` and tries to check if the rest of the pattern matches from there. ...But there's nothing in the pattern after `pattern:\d+?`. The lazy mode doesn't repeat anything without a need. The pattern finished, so we're done. We have a match `match:123 4`. -4. The next search starts from the character `5`. ```smart header="Optimizations" -Modern regular expression engines can optimize internal algorithms to work faster. So they may work a bit different from the described algorithm. +Modern regular expression engines can optimize internal algorithms to work faster. So they may work a bit differently from the described algorithm. But to understand how regular expressions work and to build regular expressions, we don't need to know about that. They are only used internally to optimize things. @@ -176,11 +175,11 @@ With regexps, there's often more than one way to do the same thing. In our case we can find quoted strings without lazy mode using the regexp `pattern:"[^"]+"`: ```js run -let reg = /"[^"]+"/g; +let regexp = /"[^"]+"/g; let str = 'a "witch" and her "broom" is one'; -alert( str.match(reg) ); // witch, broom +alert( str.match(regexp) ); // "witch", "broom" ``` The regexp `pattern:"[^"]+"` gives correct results, because it looks for a quote `pattern:'"'` followed by one or more non-quotes `pattern:[^"]`, and then the closing quote. @@ -202,20 +201,20 @@ The first idea might be: `pattern:/<a href=".*" class="doc">/g`. Let's check it: ```js run let str = '...<a href="link" class="doc">...'; -let reg = /<a href=".*" class="doc">/g; +let regexp = /<a href=".*" class="doc">/g; // Works! -alert( str.match(reg) ); // <a href="link" class="doc"> +alert( str.match(regexp) ); // <a href="link" class="doc"> ``` It worked. But let's see what happens if there are many links in the text? ```js run let str = '...<a href="link1" class="doc">... <a href="link2" class="doc">...'; -let reg = /<a href=".*" class="doc">/g; +let regexp = /<a href=".*" class="doc">/g; // Whoops! Two links in one match! -alert( str.match(reg) ); // <a href="link1" class="doc">... <a href="link2" class="doc"> +alert( str.match(regexp) ); // <a href="link1" class="doc">... <a href="link2" class="doc"> ``` Now the result is wrong for the same reason as our "witches" example. The quantifier `pattern:.*` took too many characters. @@ -231,10 +230,10 @@ Let's modify the pattern by making the quantifier `pattern:.*?` lazy: ```js run let str = '...<a href="link1" class="doc">... <a href="link2" class="doc">...'; -let reg = /<a href=".*?" class="doc">/g; +let regexp = /<a href=".*?" class="doc">/g; // Works! -alert( str.match(reg) ); // <a href="link1" class="doc">, <a href="link2" class="doc"> +alert( str.match(regexp) ); // <a href="link1" class="doc">, <a href="link2" class="doc"> ``` Now it seems to work, there are two matches: @@ -248,10 +247,10 @@ Now it seems to work, there are two matches: ```js run let str = '...<a href="link1" class="wrong">... <p style="" class="doc">...'; -let reg = /<a href=".*?" class="doc">/g; +let regexp = /<a href=".*?" class="doc">/g; // Wrong match! -alert( str.match(reg) ); // <a href="link1" class="wrong">... <p style="" class="doc"> +alert( str.match(regexp) ); // <a href="link1" class="wrong">... <p style="" class="doc"> ``` Now it fails. The match includes not just a link, but also a lot of text after it, including `<p...>`. @@ -264,7 +263,7 @@ That's what's going on: 2. Then it looks for `pattern:.*?`: takes one character (lazily!), check if there's a match for `pattern:" class="doc">` (none). 3. Then takes another character into `pattern:.*?`, and so on... until it finally reaches `match:" class="doc">`. -But the problem is: that's already beyound the link, in another tag `<p>`. Not what we want. +But the problem is: that's already beyond the link `<a...>`, in another tag `<p>`. Not what we want. Here's the picture of the match aligned with the text: @@ -273,22 +272,20 @@ Here's the picture of the match aligned with the text: <a href="link1" class="wrong">... <p style="" class="doc"> ``` -So the laziness did not work for us here. +So, we need the pattern to look for `<a href="...something..." class="doc">`, but both greedy and lazy variants have problems. -We need the pattern to look for `<a href="...something..." class="doc">`, but both greedy and lazy variants have problems. - -The correct variant would be: `pattern:href="[^"]*"`. It will take all characters inside the `href` attribute till the nearest quote, just what we need. +The correct variant can be: `pattern:href="[^"]*"`. It will take all characters inside the `href` attribute till the nearest quote, just what we need. A working example: ```js run let str1 = '...<a href="link1" class="wrong">... <p style="" class="doc">...'; let str2 = '...<a href="link1" class="doc">... <a href="link2" class="doc">...'; -let reg = /<a href="[^"]*" class="doc">/g; +let regexp = /<a href="[^"]*" class="doc">/g; // Works! -alert( str1.match(reg) ); // null, no matches, that's correct -alert( str2.match(reg) ); // <a href="link1" class="doc">, <a href="link2" class="doc"> +alert( str1.match(regexp) ); // null, no matches, that's correct +alert( str2.match(regexp) ); // <a href="link1" class="doc">, <a href="link2" class="doc"> ``` ## Summary @@ -296,9 +293,9 @@ alert( str2.match(reg) ); // <a href="link1" class="doc">, <a href="link2" class Quantifiers have two modes of work: Greedy -: By default the regular expression engine tries to repeat the quantifier as many times as possible. For instance, `pattern:\d+` consumes all possible digits. When it becomes impossible to consume more (no more digits or string end), then it continues to match the rest of the pattern. If there's no match then it decreases the number of repetitions (backtracks) and tries again. +: By default the regular expression engine tries to repeat the quantified character as many times as possible. For instance, `pattern:\d+` consumes all possible digits. When it becomes impossible to consume more (no more digits or string end), then it continues to match the rest of the pattern. If there's no match then it decreases the number of repetitions (backtracks) and tries again. Lazy -: Enabled by the question mark `pattern:?` after the quantifier. The regexp engine tries to match the rest of the pattern before each repetition of the quantifier. +: Enabled by the question mark `pattern:?` after the quantifier. The regexp engine tries to match the rest of the pattern before each repetition of the quantified character. -As we've seen, the lazy mode is not a "panacea" from the greedy search. An alternative is a "fine-tuned" greedy search, with exclusions. Soon we'll see more examples of it. +As we've seen, the lazy mode is not a "panacea" from the greedy search. An alternative is a "fine-tuned" greedy search, with exclusions, as in the pattern `pattern:"[^"]+"`. diff --git a/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy1.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy1.svg new file mode 100644 index 000000000..c9044a226 --- /dev/null +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy1.svg @@ -0,0 +1,20 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy1.svg +<?xml version="1.0" encoding="UTF-8"?> +<svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> + <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> + <title>witch_greedy1.svg</title> + <desc>Created with sketchtool.</desc> + <g id="regexp" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> + <g id="witch_greedy1.svg"> + <text id="a-"witch"-and-her-"b" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal" fill="#8A704D"> + <tspan x="20" y="112">a "witch" and her "broom" is one</tspan> + </text> + <rect id="Rectangle-1" stroke="#E8C48E" x="45.5" y="50.5" width="14" height="76"></rect> + <path d="M52.0039062,60.4414062 L51.6757812,65.9375 L49.7539062,65.9375 L49.4140625,60.4414062 L52.0039062,60.4414062 Z M56.5625,60.4414062 L56.234375,65.9375 L54.3125,65.9375 L53.9726562,60.4414062 L56.5625,60.4414062 Z" id="--"-----------------" fill="#CB1E31"></path> + <path id="Line" d="M51.5,30.5 L51.5,8.5 L53.5,8.5 L53.5,30.5 L59.5,30.5 L52.5,44.5 L45.5,30.5 L51.5,30.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> + </g> + </g> +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy1.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="112">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 50.5h14v76h-14z"/><path id="--"-----------------" fill="#A7333A" d="M52.004 60.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M53.5 8.5v22h6l-7 14-7-14h6v-22h2z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy1.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy2.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy2.svg similarity index 58% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy2.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy2.svg index 1ee351588..2575297bc 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy2.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy2.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy2.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -13,4 +14,7 @@ <path id="Line" d="M55.5,33.5 L47.5,33.5 L47.5,31.5 L55.5,31.5 L55.5,25.5 L69.5,32.5 L55.5,39.5 L55.5,33.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy2.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="107">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 45.5h28v76h-28z"/><path id="--".----------------" fill="#A7333A" d="M52.004 55.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 68.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M55.5 25.5l14 7-14 7v-6h-8v-2h8v-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy2.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy3.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy3.svg similarity index 76% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy3.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy3.svg index e60ed1618..ae122dcd0 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy3.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy3.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy3.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -13,4 +14,7 @@ <path id="Line" d="M425.5,27.5 L50.5,27.5 L50.5,25.5 L425.5,25.5 L425.5,19.5 L439.5,26.5 L425.5,33.5 L425.5,27.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy3.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="107">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 45.5h401v76h-401z"/><path id="--"................." fill="#A7333A" d="M52.004 55.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 68.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.622.422.175.18.314.388.416.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.417-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.622.422.175.18.314.388.416.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M425.5 19.5l14 7-14 7v-6h-375v-2h375v-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy3.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy4.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy4.svg similarity index 76% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy4.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy4.svg index 630a62230..6da3b01b0 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy4.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy4.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy4.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -16,4 +17,7 @@ <path d="M423.5,30.5 L439.5,30.5" id="Line" stroke="#B8BAC1" stroke-linecap="square"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy4.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="107">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 45.5h383v76h-383z"/><path id="--"................." fill="#A7333A" d="M52.004 55.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 68.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.622.422.175.18.314.388.416.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.417-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Path" fill="#478964" d="M434.668 55.441l-.328 5.496h-1.922l-.34-5.496zM439.227 55.441l-.329 5.496h-1.921l-.34-5.496z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M408.5 23.5l14 7-14 7v-6h-344v-2h344v-6z"/><path id="Line" stroke="#D1CFCD" stroke-linecap="square" d="M423.5 30.5h16"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy4.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy5.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy5.svg similarity index 76% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy5.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy5.svg index fa8db5846..173734898 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy5.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy5.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy5.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -16,4 +17,7 @@ <path d="M410.5,29.5 L435.5,29.5" id="Line" stroke="#B8BAC1" stroke-linecap="square"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy5.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="107">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 45.5h370v76h-370z"/><path id="--"................." fill="#A7333A" d="M52.004 55.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 68.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.622.422.175.18.314.388.416.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.417-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Path" fill="#478964" d="M421.473 55.441l-.328 5.496h-1.922l-.34-5.496zM426.031 55.441l-.328 5.496h-1.922l-.34-5.496z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M395.5 22.5l14 7-14 7v-6h-343v-2h343v-6z"/><path id="Line" stroke="#D1CFCD" stroke-linecap="square" d="M410.5 29.5h25"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy5.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy6.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy6.svg similarity index 75% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy6.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy6.svg index c0165e88d..bf9557612 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy6.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy6.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_greedy6.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -14,4 +15,7 @@ <path d="M346.5,30.5 L437.5,30.5" id="Line" stroke="#B8BAC1" stroke-linecap="square"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_greedy6.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="107">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 45.5h305v76h-305z"/><path id="--"................." fill="#A7333A" d="M52.004 55.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 68.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.446.242.621.422c.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.176.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.62.422c.177.18.315.388.417.627.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.622.422.175.18.314.388.416.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.032-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.755.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.417-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm12.305-12.868l-.328 5.496h-1.922l-.34-5.496h2.59zm4.558 0l-.328 5.496h-1.922l-.34-5.496h2.59z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M331.5 23.5l14 7-14 7v-6h-281v-2h281v-6z"/><path id="Line" stroke="#D1CFCD" stroke-linecap="square" d="M346.5 30.5h91"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_greedy6.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy3.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy3.svg similarity index 59% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy3.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy3.svg index 61506bb5a..c12fb7289 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy3.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy3.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy3.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -15,4 +16,7 @@ <path id="Line" d="M60,34.5 L45,34.5 L45,32.5 L60,32.5 L60,26.5 L74,33.5 L60,40.5 L60,34.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_lazy3.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="105">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 43.5h28v76h-28z"/><path id="--"."---------------" fill="#A7333A" d="M52.004 53.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 66.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Path" fill="#478964" d="M78.395 53.441l-.329 5.496h-1.921l-.34-5.496zM82.953 53.441l-.328 5.496h-1.922l-.34-5.496z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M60 26.5l14 7-14 7v-6H45v-2h15v-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy3.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy4.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy4.svg similarity index 64% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy4.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy4.svg index fecc1a5ae..353f8f447 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy4.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy4.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy4.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -15,4 +16,7 @@ <path id="Line" d="M70,30.5 L46,30.5 L46,28.5 L70,28.5 L70,22.5 L84,29.5 L70,36.5 L70,30.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_lazy4.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="105">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 43.5h40v76h-40z"/><path id="--".."--------------" fill="#A7333A" d="M52.004 53.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 66.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574z"/><path id="Path" fill="#478964" d="M91.59 53.441l-.328 5.496H89.34L89 53.441zM96.148 53.441l-.328 5.496h-1.922l-.34-5.496z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M70 22.5l14 7-14 7-.001-6H46v-2h23.999l.001-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy4.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy5.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy5.svg similarity index 70% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy5.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy5.svg index b75a9d7bd..cbd3a1dc7 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy5.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy5.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy5.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -13,4 +14,7 @@ <path id="Line" d="M120.5,31.5 L51.5,31.5 L51.5,29.5 L120.5,29.5 L120.5,23.5 L134.5,30.5 L120.5,37.5 L120.5,31.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_lazy5.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="105">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 43.5h94v76h-94z"/><path id="--"....."---------"." fill="#A7333A" d="M52.004 53.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 66.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm12.305-12.868l-.328 5.496h-1.922l-.34-5.496h2.59zm4.558 0l-.328 5.496h-1.922l-.34-5.496h2.59z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M120.5 23.5l14 7-14 7v-6h-69v-2h69v-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy5.svg diff --git a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy6.svg b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy6.svg similarity index 73% rename from 9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy6.svg rename to 9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy6.svg index 867e30c85..0bf09cde3 100644 --- a/9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy6.svg +++ b/9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy6.svg @@ -1,3 +1,4 @@ +<<<<<<< HEAD:9-regular-expressions/08-regexp-greedy-and-lazy/witch_lazy6.svg <?xml version="1.0" encoding="UTF-8"?> <svg width="463px" height="130px" viewBox="0 0 463 130" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> <!-- Generator: sketchtool 55.2 (78181) - https://sketchapp.com --> @@ -15,4 +16,7 @@ <path id="Line" d="M118.5,35.5 L53.5,35.5 L53.5,33.5 L118.5,33.5 L118.5,27.5 L132.5,34.5 L118.5,41.5 L118.5,35.5 Z" fill="#EE6B47" fill-rule="nonzero"></path> </g> </g> -</svg> \ No newline at end of file +</svg> +======= +<svg xmlns="http://www.w3.org/2000/svg" width="463" height="130" viewBox="0 0 463 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="witch_lazy6.svg"><text id="a-"witch"-and-her-"b" fill="#AF6E24" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="105">a "witch" and her "broom" is one</tspan></text><path id="Rectangle-1" stroke="#DBAF88" d="M45.5 43.5h94v76h-94z"/><path id="Rectangle-2" stroke="#DBAF88" d="M256.5 43.5h94v76h-94z"/><path id="--"....."---------"." fill="#A7333A" d="M52.004 53.441l-.328 5.496h-1.922l-.34-5.496h2.59zm4.559 0l-.329 5.496h-1.922l-.34-5.496h2.59zM66.09 66.31c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm12.305-12.868l-.328 5.496h-1.922l-.34-5.496h2.59zm4.558 0l-.328 5.496h-1.922l-.34-5.496h2.59zm127.395 0l-.328 5.496h-1.922l-.34-5.496h2.59zm4.558 0l-.328 5.496h-1.921l-.34-5.496h2.59zm9.528 12.868c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.527-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.152-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.315.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152.239.102.446.242.621.422.176.18.315.388.416.627.102.238.153.494.153.767 0 .266-.051.516-.153.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.761-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.417-.627 1.895 1.895 0 011.377-.574zm13.196 0c.265 0 .517.05.756.152.238.102.445.242.62.422.177.18.315.388.417.627.101.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.274 0-.528-.05-.762-.152a1.967 1.967 0 01-1.031-1.032 1.866 1.866 0 01-.153-.75c0-.273.051-.529.153-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm13.195 0c.266 0 .518.05.756.152s.445.242.621.422.314.388.416.627c.102.238.152.494.152.767 0 .266-.05.516-.152.75a1.95 1.95 0 01-1.793 1.184c-.273 0-.527-.05-.762-.152a1.967 1.967 0 01-1.03-1.032 1.866 1.866 0 01-.153-.75c0-.273.05-.529.152-.767a2.02 2.02 0 01.416-.627 1.895 1.895 0 011.377-.574zm12.305-12.868l-.328 5.496h-1.922l-.34-5.496h2.59zm4.558 0l-.328 5.496h-1.922l-.34-5.496h2.59z"/><path id="Line" fill="#C06334" fill-rule="nonzero" d="M331.5 27.5l14 7-14 7v-6h-70v-2h70v-6zM118.5 27.5l14 7-14 7v-6h-65v-2h65v-6z"/></g></g></svg> +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b:9-regular-expressions/10-regexp-greedy-and-lazy/witch_lazy6.svg diff --git a/9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/solution.md b/9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/solution.md deleted file mode 100644 index 03080f86c..000000000 --- a/9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/solution.md +++ /dev/null @@ -1,23 +0,0 @@ - -Opening tag is `pattern:\[(b|url|quote)\]`. - -Then to find everything till the closing tag -- let's the pattern `pattern:[\s\S]*?` to match any character including the newline and then a backreference to the closing tag. - -The full pattern: `pattern:\[(b|url|quote)\][\s\S]*?\[/\1\]`. - -In action: - -```js run -let reg = /\[(b|url|quote)\][\s\S]*?\[\/\1\]/g; - -let str = ` - [b]hello![/b] - [quote] - [url]http://google.com[/url] - [/quote] -`; - -alert( str.match(reg) ); // [b]hello![/b],[quote][url]http://google.com[/url][/quote] -``` - -Please note that we had to escape a slash for the closing tag `pattern:[/\1]`, because normally the slash closes the pattern. diff --git a/9-regular-expressions/11-regexp-alternation/03-match-quoted-string/task.md b/9-regular-expressions/11-regexp-alternation/03-match-quoted-string/task.md deleted file mode 100644 index 2ccac4bdf..000000000 --- a/9-regular-expressions/11-regexp-alternation/03-match-quoted-string/task.md +++ /dev/null @@ -1,32 +0,0 @@ -# Find quoted strings - -Create a regexp to find strings in double quotes `subject:"..."`. - -The important part is that strings should support escaping, in the same way as JavaScript strings do. For instance, quotes can be inserted as `subject:\"` a newline as `subject:\n`, and the slash itself as `subject:\\`. - -```js -let str = "Just like \"here\"."; -``` - -For us it's important that an escaped quote `subject:\"` does not end a string. - -So we should look from one quote to the other ignoring escaped quotes on the way. - -That's the essential part of the task, otherwise it would be trivial. - -Examples of strings to match: -```js -.. *!*"test me"*/!* .. -.. *!*"Say \"Hello\"!"*/!* ... (escaped quotes inside) -.. *!*"\\"*/!* .. (double slash inside) -.. *!*"\\ \""*/!* .. (double slash and an escaped quote inside) -``` - -In JavaScript we need to double the slashes to pass them right into the string, like this: - -```js run -let str = ' .. "test me" .. "Say \\"Hello\\"!" .. "\\\\ \\"" .. '; - -// the in-memory string -alert(str); // .. "test me" .. "Say \"Hello\"!" .. "\\ \"" .. -``` diff --git a/9-regular-expressions/11-regexp-alternation/article.md b/9-regular-expressions/11-regexp-alternation/article.md deleted file mode 100644 index 0caa6de02..000000000 --- a/9-regular-expressions/11-regexp-alternation/article.md +++ /dev/null @@ -1,59 +0,0 @@ -# Alternation (OR) | - -Alternation is the term in regular expression that is actually a simple "OR". - -In a regular expression it is denoted with a vertical line character `pattern:|`. - -For instance, we need to find programming languages: HTML, PHP, Java or JavaScript. - -The corresponding regexp: `pattern:html|php|java(script)?`. - -A usage example: - -```js run -let reg = /html|php|css|java(script)?/gi; - -let str = "First HTML appeared, then CSS, then JavaScript"; - -alert( str.match(reg) ); // 'HTML', 'CSS', 'JavaScript' -``` - -We already know a similar thing -- square brackets. They allow to choose between multiple character, for instance `pattern:gr[ae]y` matches `match:gray` or `match:grey`. - -Square brackets allow only characters or character sets. Alternation allows any expressions. A regexp `pattern:A|B|C` means one of expressions `A`, `B` or `C`. - -For instance: - -- `pattern:gr(a|e)y` means exactly the same as `pattern:gr[ae]y`. -- `pattern:gra|ey` means `match:gra` or `match:ey`. - -To separate a part of the pattern for alternation we usually enclose it in parentheses, like this: `pattern:before(XXX|YYY)after`. - -## Regexp for time - -In previous chapters there was a task to build a regexp for searching time in the form `hh:mm`, for instance `12:00`. But a simple `pattern:\d\d:\d\d` is too vague. It accepts `25:99` as the time (99 seconds is valid, but shouldn't be). - -How can we make a better one? - -We can apply more careful matching. First, the hours: - -- If the first digit is `0` or `1`, then the next digit can by anything. -- Or, if the first digit is `2`, then the next must be `pattern:[0-3]`. - -As a regexp: `pattern:[01]\d|2[0-3]`. - -Next, the minutes must be from `0` to `59`. In the regexp language that means `pattern:[0-5]\d`: the first digit `0-5`, and then any digit. - -Let's glue them together into the pattern: `pattern:[01]\d|2[0-3]:[0-5]\d`. - -We're almost done, but there's a problem. The alternation `pattern:|` now happens to be between `pattern:[01]\d` and `pattern:2[0-3]:[0-5]\d`. - -That's wrong, as it should be applied only to hours `[01]\d` OR `2[0-3]`. That's a common mistake when starting to work with regular expressions. - -The correct variant: - -```js run -let reg = /([01]\d|2[0-3]):[0-5]\d/g; - -alert("00:00 10:10 23:59 25:99 1:2".match(reg)); // 00:00,10:10,23:59 -``` diff --git a/9-regular-expressions/11-regexp-groups/01-test-mac/solution.md b/9-regular-expressions/11-regexp-groups/01-test-mac/solution.md new file mode 100644 index 000000000..f7a5f1e39 --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/01-test-mac/solution.md @@ -0,0 +1,21 @@ +A two-digit hex number is `pattern:[0-9a-f]{2}` (assuming the flag `pattern:i` is set). + +We need that number `NN`, and then `:NN` repeated 5 times (more numbers); + +The regexp is: `pattern:[0-9a-f]{2}(:[0-9a-f]{2}){5}` + +Now let's show that the match should capture all the text: start at the beginning and end at the end. That's done by wrapping the pattern in `pattern:^...$`. + +Finally: + +```js run +let regexp = /^[0-9a-f]{2}(:[0-9a-f]{2}){5}$/i; + +alert( regexp.test('01:32:54:67:89:AB') ); // true + +alert( regexp.test('0132546789AB') ); // false (no colons) + +alert( regexp.test('01:32:54:67:89') ); // false (5 numbers, need 6) + +alert( regexp.test('01:32:54:67:89:ZZ') ) // false (ZZ in the end) +``` diff --git a/9-regular-expressions/12-regexp-anchors/2-test-mac/task.md b/9-regular-expressions/11-regexp-groups/01-test-mac/task.md similarity index 50% rename from 9-regular-expressions/12-regexp-anchors/2-test-mac/task.md rename to 9-regular-expressions/11-regexp-groups/01-test-mac/task.md index e72655984..a2e799cfa 100644 --- a/9-regular-expressions/12-regexp-anchors/2-test-mac/task.md +++ b/9-regular-expressions/11-regexp-groups/01-test-mac/task.md @@ -8,13 +8,13 @@ Write a regexp that checks whether a string is MAC-address. Usage: ```js -let reg = /your regexp/; +let regexp = /your regexp/; -alert( reg.test('01:32:54:67:89:AB') ); // true +alert( regexp.test('01:32:54:67:89:AB') ); // true -alert( reg.test('0132546789AB') ); // false (no colons) +alert( regexp.test('0132546789AB') ); // false (no colons) -alert( reg.test('01:32:54:67:89') ); // false (5 numbers, must be 6) +alert( regexp.test('01:32:54:67:89') ); // false (5 numbers, must be 6) -alert( reg.test('01:32:54:67:89:ZZ') ) // false (ZZ ad the end) +alert( regexp.test('01:32:54:67:89:ZZ') ) // false (ZZ at the end) ``` diff --git a/9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/solution.md b/9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/solution.md new file mode 100644 index 000000000..0806dc4fd --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/solution.md @@ -0,0 +1,27 @@ +A regexp to search 3-digit color `#abc`: `pattern:/#[a-f0-9]{3}/i`. + +We can add exactly 3 more optional hex digits. We don't need more or less. The color has either 3 or 6 digits. + +Let's use the quantifier `pattern:{1,2}` for that: we'll have `pattern:/#([a-f0-9]{3}){1,2}/i`. + +Here the pattern `pattern:[a-f0-9]{3}` is enclosed in parentheses to apply the quantifier `pattern:{1,2}`. + +In action: + +```js run +let regexp = /#([a-f0-9]{3}){1,2}/gi; + +let str = "color: #3f3; background-color: #AA00ef; and: #abcd"; + +alert( str.match(regexp) ); // #3f3 #AA00ef #abc +``` + +There's a minor problem here: the pattern found `match:#abc` in `subject:#abcd`. To prevent that we can add `pattern:\b` to the end: + +```js run +let regexp = /#([a-f0-9]{3}){1,2}\b/gi; + +let str = "color: #3f3; background-color: #AA00ef; and: #abcd"; + +alert( str.match(regexp) ); // #3f3 #AA00ef +``` diff --git a/9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/task.md b/9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/task.md similarity index 59% rename from 9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/task.md rename to 9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/task.md index 4efd6f61f..09108484a 100644 --- a/9-regular-expressions/09-regexp-groups/1-find-webcolor-3-or-6/task.md +++ b/9-regular-expressions/11-regexp-groups/02-find-webcolor-3-or-6/task.md @@ -4,11 +4,11 @@ Write a RegExp that matches colors in the format `#abc` or `#abcdef`. That is: ` Usage example: ```js -let reg = /your regexp/g; +let regexp = /your regexp/g; let str = "color: #3f3; background-color: #AA00ef; and: #abcd"; -alert( str.match(reg) ); // #3f3 #AA00ef +alert( str.match(regexp) ); // #3f3 #AA00ef ``` -P.S. This should be exactly 3 or 6 hex digits: values like `#abcd` should not match. +P.S. This should be exactly 3 or 6 hex digits. Values with 4 digits, such as `#abcd`, should not match. diff --git a/9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/solution.md b/9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/solution.md new file mode 100644 index 000000000..813d619ef --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/solution.md @@ -0,0 +1,11 @@ +A positive number with an optional decimal part is: `pattern:\d+(\.\d+)?`. + +Let's add the optional `pattern:-` in the beginning: + +```js run +let regexp = /-?\d+(\.\d+)?/g; + +let str = "-1.5 0 2 -123.4."; + +alert( str.match(regexp) ); // -1.5, 0, 2, -123.4 +``` diff --git a/9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/task.md b/9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/task.md similarity index 71% rename from 9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/task.md rename to 9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/task.md index 121a18a41..4f5a73fff 100644 --- a/9-regular-expressions/09-regexp-groups/4-find-decimal-numbers/task.md +++ b/9-regular-expressions/11-regexp-groups/03-find-decimal-numbers/task.md @@ -5,9 +5,9 @@ Write a regexp that looks for all decimal numbers including integer ones, with t An example of use: ```js -let reg = /your regexp/g; +let regexp = /your regexp/g; let str = "-1.5 0 2 -123.4."; -alert( str.match(re) ); // -1.5, 0, 2, -123.4 +alert( str.match(regexp) ); // -1.5, 0, 2, -123.4 ``` diff --git a/9-regular-expressions/11-regexp-groups/04-parse-expression/solution.md b/9-regular-expressions/11-regexp-groups/04-parse-expression/solution.md new file mode 100644 index 000000000..b23813395 --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/04-parse-expression/solution.md @@ -0,0 +1,70 @@ +A regexp for a number is: `pattern:-?\d+(\.\d+)?`. We created it in the previous task. + +An operator is `pattern:[-+*/]`. The hyphen `pattern:-` goes first in the square brackets, because in the middle it would mean a character range, while we just want a character `-`. + +The slash `/` should be escaped inside a JavaScript regexp `pattern:/.../`, we'll do that later. + +We need a number, an operator, and then another number. And optional spaces between them. + +The full regular expression: `pattern:-?\d+(\.\d+)?\s*[-+*/]\s*-?\d+(\.\d+)?`. + +It has 3 parts, with `pattern:\s*` between them: +1. `pattern:-?\d+(\.\d+)?` - the first number, +2. `pattern:[-+*/]` - the operator, +3. `pattern:-?\d+(\.\d+)?` - the second number. + +To make each of these parts a separate element of the result array, let's enclose them in parentheses: `pattern:(-?\d+(\.\d+)?)\s*([-+*/])\s*(-?\d+(\.\d+)?)`. + +In action: + +```js run +let regexp = /(-?\d+(\.\d+)?)\s*([-+*\/])\s*(-?\d+(\.\d+)?)/; + +alert( "1.2 + 12".match(regexp) ); +``` + +The result includes: + +- `result[0] == "1.2 + 12"` (full match) +- `result[1] == "1.2"` (first group `(-?\d+(\.\d+)?)` -- the first number, including the decimal part) +- `result[2] == ".2"` (second group`(\.\d+)?` -- the first decimal part) +- `result[3] == "+"` (third group `([-+*\/])` -- the operator) +- `result[4] == "12"` (forth group `(-?\d+(\.\d+)?)` -- the second number) +- `result[5] == undefined` (fifth group `(\.\d+)?` -- the last decimal part is absent, so it's undefined) + +We only want the numbers and the operator, without the full match or the decimal parts, so let's "clean" the result a bit. + +The full match (the arrays first item) can be removed by shifting the array `result.shift()`. + +Groups that contain decimal parts (number 2 and 4) `pattern:(.\d+)` can be excluded by adding `pattern:?:` to the beginning: `pattern:(?:\.\d+)?`. + +The final solution: + +```js run +function parse(expr) { + let regexp = /(-?\d+(?:\.\d+)?)\s*([-+*\/])\s*(-?\d+(?:\.\d+)?)/; + + let result = expr.match(regexp); + + if (!result) return []; + result.shift(); + + return result; +} + +alert( parse("-1.23 * 3.45") ); // -1.23, *, 3.45 +``` + +As an alternative to using the non-capturing `?:`, we could name the groups, like this: + +```js run +function parse(expr) { + let regexp = /(?<a>-?\d+(?:\.\d+)?)\s*(?<operator>[-+*\/])\s*(?<b>-?\d+(?:\.\d+)?)/; + + let result = expr.match(regexp); + + return [result.groups.a, result.groups.operator, result.groups.b]; +} + +alert( parse("-1.23 * 3.45") ); // -1.23, *, 3.45; +``` \ No newline at end of file diff --git a/9-regular-expressions/09-regexp-groups/5-parse-expression/task.md b/9-regular-expressions/11-regexp-groups/04-parse-expression/task.md similarity index 100% rename from 9-regular-expressions/09-regexp-groups/5-parse-expression/task.md rename to 9-regular-expressions/11-regexp-groups/04-parse-expression/task.md diff --git a/9-regular-expressions/11-regexp-groups/article.md b/9-regular-expressions/11-regexp-groups/article.md new file mode 100644 index 000000000..8fec60ccc --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/article.md @@ -0,0 +1,364 @@ +# Capturing groups + +A part of a pattern can be enclosed in parentheses `pattern:(...)`. This is called a "capturing group". + +That has two effects: + +1. It allows to get a part of the match as a separate item in the result array. +2. If we put a quantifier after the parentheses, it applies to the parentheses as a whole. + +## Examples + +Let's see how parentheses work in examples. + +### Example: gogogo + +Without parentheses, the pattern `pattern:go+` means `subject:g` character, followed by `subject:o` repeated one or more times. For instance, `match:goooo` or `match:gooooooooo`. + +Parentheses group characters together, so `pattern:(go)+` means `match:go`, `match:gogo`, `match:gogogo` and so on. + +```js run +alert( 'Gogogo now!'.match(/(go)+/ig) ); // "Gogogo" +``` + +### Example: domain + +Let's make something more complex -- a regular expression to search for a website domain. + +For example: + +``` +mail.com +users.mail.com +smith.users.mail.com +``` + +As we can see, a domain consists of repeated words, a dot after each one except the last one. + +In regular expressions that's `pattern:(\w+\.)+\w+`: + +```js run +let regexp = /(\w+\.)+\w+/g; + +alert( "site.com my.site.com".match(regexp) ); // site.com,my.site.com +``` + +The search works, but the pattern can't match a domain with a hyphen, e.g. `my-site.com`, because the hyphen does not belong to class `pattern:\w`. + +We can fix it by replacing `pattern:\w` with `pattern:[\w-]` in every word except the last one: `pattern:([\w-]+\.)+\w+`. + +### Example: email + +The previous example can be extended. We can create a regular expression for emails based on it. + +The email format is: `name@domain`. Any word can be the name, hyphens and dots are allowed. In regular expressions that's `pattern:[-.\w]+`. + +The pattern: + +```js run +let regexp = /[-.\w]+@([\w-]+\.)+[\w-]+/g; + +alert("my@mail.com @ his@site.com.uk".match(regexp)); // my@mail.com, his@site.com.uk +``` + +That regexp is not perfect, but mostly works and helps to fix accidental mistypes. The only truly reliable check for an email can only be done by sending a letter. + +## Parentheses contents in the match + +Parentheses are numbered from left to right. The search engine memorizes the content matched by each of them and allows to get it in the result. + +The method `str.match(regexp)`, if `regexp` has no flag `g`, looks for the first match and returns it as an array: + +1. At index `0`: the full match. +2. At index `1`: the contents of the first parentheses. +3. At index `2`: the contents of the second parentheses. +4. ...and so on... + +For instance, we'd like to find HTML tags `pattern:<.*?>`, and process them. It would be convenient to have tag content (what's inside the angles), in a separate variable. + +Let's wrap the inner content into parentheses, like this: `pattern:<(.*?)>`. + +Now we'll get both the tag as a whole `match:<h1>` and its contents `match:h1` in the resulting array: + +```js run +let str = '<h1>Hello, world!</h1>'; + +let tag = str.match(/<(.*?)>/); + +alert( tag[0] ); // <h1> +alert( tag[1] ); // h1 +``` + +### Nested groups + +Parentheses can be nested. In this case the numbering also goes from left to right. + +For instance, when searching a tag in `subject:<span class="my">` we may be interested in: + +1. The tag content as a whole: `match:span class="my"`. +2. The tag name: `match:span`. +3. The tag attributes: `match:class="my"`. + +Let's add parentheses for them: `pattern:<(([a-z]+)\s*([^>]*))>`. + +Here's how they are numbered (left to right, by the opening paren): + + + +In action: + +```js run +let str = '<span class="my">'; + +let regexp = /<(([a-z]+)\s*([^>]*))>/; + +let result = str.match(regexp); +alert(result[0]); // <span class="my"> +alert(result[1]); // span class="my" +alert(result[2]); // span +alert(result[3]); // class="my" +``` + +The zero index of `result` always holds the full match. + +Then groups, numbered from left to right by an opening paren. The first group is returned as `result[1]`. Here it encloses the whole tag content. + +Then in `result[2]` goes the group from the second opening paren `pattern:([a-z]+)` - tag name, then in `result[3]` the tag: `pattern:([^>]*)`. + +The contents of every group in the string: + + + +### Optional groups + +Even if a group is optional and doesn't exist in the match (e.g. has the quantifier `pattern:(...)?`), the corresponding `result` array item is present and equals `undefined`. + +For instance, let's consider the regexp `pattern:a(z)?(c)?`. It looks for `"a"` optionally followed by `"z"` optionally followed by `"c"`. + +If we run it on the string with a single letter `subject:a`, then the result is: + +```js run +let match = 'a'.match(/a(z)?(c)?/); + +alert( match.length ); // 3 +alert( match[0] ); // a (whole match) +alert( match[1] ); // undefined +alert( match[2] ); // undefined +``` + +The array has the length of `3`, but all groups are empty. + +And here's a more complex match for the string `subject:ac`: + +```js run +let match = 'ac'.match(/a(z)?(c)?/) + +alert( match.length ); // 3 +alert( match[0] ); // ac (whole match) +alert( match[1] ); // undefined, because there's nothing for (z)? +alert( match[2] ); // c +``` + +The array length is permanent: `3`. But there's nothing for the group `pattern:(z)?`, so the result is `["ac", undefined, "c"]`. + +## Searching for all matches with groups: matchAll + +```warn header="`matchAll` is a new method, polyfill may be needed" +The method `matchAll` is not supported in old browsers. + +A polyfill may be required, such as <https://github.com/ljharb/String.prototype.matchAll>. +``` + +When we search for all matches (flag `pattern:g`), the `match` method does not return contents for groups. + +For example, let's find all tags in a string: + +```js run +let str = '<h1> <h2>'; + +let tags = str.match(/<(.*?)>/g); + +alert( tags ); // <h1>,<h2> +``` + +The result is an array of matches, but without details about each of them. But in practice we usually need contents of capturing groups in the result. + +To get them, we should search using the method `str.matchAll(regexp)`. + +It was added to JavaScript language long after `match`, as its "new and improved version". + +Just like `match`, it looks for matches, but there are 3 differences: + +1. It returns not an array, but an iterable object. +2. When the flag `pattern:g` is present, it returns every match as an array with groups. +3. If there are no matches, it returns not `null`, but an empty iterable object. + +For instance: + +```js run +let results = '<h1> <h2>'.matchAll(/<(.*?)>/gi); + +// results - is not an array, but an iterable object +alert(results); // [object RegExp String Iterator] + +alert(results[0]); // undefined (*) + +results = Array.from(results); // let's turn it into array + +alert(results[0]); // <h1>,h1 (1st tag) +alert(results[1]); // <h2>,h2 (2nd tag) +``` + +As we can see, the first difference is very important, as demonstrated in the line `(*)`. We can't get the match as `results[0]`, because that object is a pseudoarray. We can turn it into a real `Array` using `Array.from`. There are more details about pseudoarrays and iterables in the article <info:iterable>. + +There's no need for `Array.from` if we're looping over results: + +```js run +let results = '<h1> <h2>'.matchAll(/<(.*?)>/gi); + +for(let result of results) { + alert(result); + // first alert: <h1>,h1 + // second: <h2>,h2 +} +``` + +...Or using destructuring: + +```js +let [tag1, tag2] = '<h1> <h2>'.matchAll(/<(.*?)>/gi); +``` + +Every match, returned by `matchAll`, has the same format as returned by `match` without flag `pattern:g`: it's an array with additional properties `index` (match index in the string) and `input` (source string): + +```js run +let results = '<h1> <h2>'.matchAll(/<(.*?)>/gi); + +let [tag1, tag2] = results; + +alert( tag1[0] ); // <h1> +alert( tag1[1] ); // h1 +alert( tag1.index ); // 0 +alert( tag1.input ); // <h1> <h2> +``` + +```smart header="Why is a result of `matchAll` an iterable object, not an array?" +Why is the method designed like that? The reason is simple - for the optimization. + +The call to `matchAll` does not perform the search. Instead, it returns an iterable object, without the results initially. The search is performed each time we iterate over it, e.g. in the loop. + +So, there will be found as many results as needed, not more. + +E.g. there are potentially 100 matches in the text, but in a `for..of` loop we found 5 of them, then decided it's enough and made a `break`. Then the engine won't spend time finding other 95 matches. +``` + +## Named groups + +Remembering groups by their numbers is hard. For simple patterns it's doable, but for more complex ones counting parentheses is inconvenient. We have a much better option: give names to parentheses. + +That's done by putting `pattern:?<name>` immediately after the opening paren. + +For example, let's look for a date in the format "year-month-day": + +```js run +*!* +let dateRegexp = /(?<year>[0-9]{4})-(?<month>[0-9]{2})-(?<day>[0-9]{2})/; +*/!* +let str = "2019-04-30"; + +let groups = str.match(dateRegexp).groups; + +alert(groups.year); // 2019 +alert(groups.month); // 04 +alert(groups.day); // 30 +``` + +As you can see, the groups reside in the `.groups` property of the match. + +To look for all dates, we can add flag `pattern:g`. + +We'll also need `matchAll` to obtain full matches, together with groups: + +```js run +let dateRegexp = /(?<year>[0-9]{4})-(?<month>[0-9]{2})-(?<day>[0-9]{2})/g; + +let str = "2019-10-30 2020-01-01"; + +let results = str.matchAll(dateRegexp); + +for(let result of results) { + let {year, month, day} = result.groups; + + alert(`${day}.${month}.${year}`); + // first alert: 30.10.2019 + // second: 01.01.2020 +} +``` + +## Capturing groups in replacement + +Method `str.replace(regexp, replacement)` that replaces all matches with `regexp` in `str` allows to use parentheses contents in the `replacement` string. That's done using `pattern:$n`, where `pattern:n` is the group number. + +For example, + +```js run +let str = "John Bull"; +let regexp = /(\w+) (\w+)/; + +alert( str.replace(regexp, '$2, $1') ); // Bull, John +``` + +For named parentheses the reference will be `pattern:$<name>`. + +For example, let's reformat dates from "year-month-day" to "day.month.year": + +```js run +let regexp = /(?<year>[0-9]{4})-(?<month>[0-9]{2})-(?<day>[0-9]{2})/g; + +let str = "2019-10-30, 2020-01-01"; + +alert( str.replace(regexp, '$<day>.$<month>.$<year>') ); +// 30.10.2019, 01.01.2020 +``` + +## Non-capturing groups with ?: + +Sometimes we need parentheses to correctly apply a quantifier, but we don't want their contents in results. + +A group may be excluded by adding `pattern:?:` in the beginning. + +For instance, if we want to find `pattern:(go)+`, but don't want the parentheses contents (`go`) as a separate array item, we can write: `pattern:(?:go)+`. + +In the example below we only get the name `match:John` as a separate member of the match: + +```js run +let str = "Gogogo John!"; + +*!* +// ?: excludes 'go' from capturing +let regexp = /(?:go)+ (\w+)/i; +*/!* + +let result = str.match(regexp); + +alert( result[0] ); // Gogogo John (full match) +alert( result[1] ); // John +alert( result.length ); // 2 (no more items in the array) +``` + +## Summary + +Parentheses group together a part of the regular expression, so that the quantifier applies to it as a whole. + +Parentheses groups are numbered left-to-right, and can optionally be named with `(?<name>...)`. + +The content, matched by a group, can be obtained in the results: + +- The method `str.match` returns capturing groups only without flag `pattern:g`. +- The method `str.matchAll` always returns capturing groups. + +If the parentheses have no name, then their contents is available in the match array by its number. Named parentheses are also available in the property `groups`. + +We can also use parentheses contents in the replacement string in `str.replace`: by the number `$n` or the name `$<name>`. + +A group may be excluded from numbering by adding `pattern:?:` in its start. That's used when we need to apply a quantifier to the whole group, but don't want it as a separate item in the results array. We also can't reference such parentheses in the replacement string. diff --git a/9-regular-expressions/11-regexp-groups/regexp-nested-groups-matches.svg b/9-regular-expressions/11-regexp-groups/regexp-nested-groups-matches.svg new file mode 100644 index 000000000..ce61ff3a7 --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/regexp-nested-groups-matches.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="320" height="130" viewBox="0 0 320 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="regexp-nested-groups.svg"><text id="<(([a-z]+)\s*([^>]*)" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="75" fill="#8A704D"><</tspan> <tspan x="33.2" y="75" fill="#DB2023">((</tspan> <tspan x="59.6" y="75" fill="#8A704D">[a-z]+</tspan> <tspan x="138.8" y="75" fill="#DB2023">)</tspan> <tspan x="152" y="75" fill="#8A704D">\s*</tspan> <tspan x="191.6" y="75" fill="#DB2023">(</tspan> <tspan x="204.8" y="75" fill="#8A704D">[^>]*</tspan> <tspan x="270.8" y="75" fill="#D0021B">))</tspan> <tspan x="297.2" y="75" fill="#8A704D">></tspan></text><path id="Line" stroke="#D0021B" stroke-linecap="square" d="M42.5 45.646V29.354"/><path id="Line-2" stroke="#D0021B" stroke-linecap="square" d="M290.5 45.646V29.354"/><path id="Line" stroke="#D0021B" stroke-linecap="square" d="M42.5 28.5h248"/><path id="Line-5" stroke="#D0021B" stroke-linecap="square" d="M52.5 101.646V85.354"/><path id="Line-4" stroke="#D0021B" stroke-linecap="square" d="M145.5 101.646V85.354"/><path id="Line-3" stroke="#D0021B" stroke-linecap="square" d="M52.5 102.5h93"/><text id="1" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="24" y="44">1</tspan></text><text id="span-class="my"" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="82" y="23">span class="my"</tspan></text><text id="2" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="35" y="101">2</tspan></text><text id="span" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="73" y="119">span</tspan></text><path id="Line-8" stroke="#D0021B" stroke-linecap="square" d="M197.5 101.646V85.354"/><path id="Line-7" stroke="#D0021B" stroke-linecap="square" d="M277.5 101.646V85.354"/><path id="Line-6" stroke="#D0021B" stroke-linecap="square" d="M197.5 102.5h80"/><text id="3" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="182" y="101">3</tspan></text><text id="class="my"" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="185" y="121">class="my"</tspan></text></g></g></svg> \ No newline at end of file diff --git a/9-regular-expressions/11-regexp-groups/regexp-nested-groups-pattern.svg b/9-regular-expressions/11-regexp-groups/regexp-nested-groups-pattern.svg new file mode 100644 index 000000000..ce61ff3a7 --- /dev/null +++ b/9-regular-expressions/11-regexp-groups/regexp-nested-groups-pattern.svg @@ -0,0 +1 @@ +<svg xmlns="http://www.w3.org/2000/svg" width="320" height="130" viewBox="0 0 320 130"><defs><style>@import url(https://fonts.googleapis.com/css?family=Open+Sans:bold,italic,bolditalic%7CPT+Mono);@font-face{font-family:'PT Mono';font-weight:700;font-style:normal;src:local('PT MonoBold'),url(/font/PTMonoBold.woff2) format('woff2'),url(/font/PTMonoBold.woff) format('woff'),url(/font/PTMonoBold.ttf) format('truetype')}</style></defs><g id="regexp" fill="none" fill-rule="evenodd" stroke="none" stroke-width="1"><g id="regexp-nested-groups.svg"><text id="<(([a-z]+)\s*([^>]*)" font-family="PTMono-Regular, PT Mono" font-size="22" font-weight="normal"><tspan x="20" y="75" fill="#8A704D"><</tspan> <tspan x="33.2" y="75" fill="#DB2023">((</tspan> <tspan x="59.6" y="75" fill="#8A704D">[a-z]+</tspan> <tspan x="138.8" y="75" fill="#DB2023">)</tspan> <tspan x="152" y="75" fill="#8A704D">\s*</tspan> <tspan x="191.6" y="75" fill="#DB2023">(</tspan> <tspan x="204.8" y="75" fill="#8A704D">[^>]*</tspan> <tspan x="270.8" y="75" fill="#D0021B">))</tspan> <tspan x="297.2" y="75" fill="#8A704D">></tspan></text><path id="Line" stroke="#D0021B" stroke-linecap="square" d="M42.5 45.646V29.354"/><path id="Line-2" stroke="#D0021B" stroke-linecap="square" d="M290.5 45.646V29.354"/><path id="Line" stroke="#D0021B" stroke-linecap="square" d="M42.5 28.5h248"/><path id="Line-5" stroke="#D0021B" stroke-linecap="square" d="M52.5 101.646V85.354"/><path id="Line-4" stroke="#D0021B" stroke-linecap="square" d="M145.5 101.646V85.354"/><path id="Line-3" stroke="#D0021B" stroke-linecap="square" d="M52.5 102.5h93"/><text id="1" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="24" y="44">1</tspan></text><text id="span-class="my"" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="82" y="23">span class="my"</tspan></text><text id="2" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="35" y="101">2</tspan></text><text id="span" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="73" y="119">span</tspan></text><path id="Line-8" stroke="#D0021B" stroke-linecap="square" d="M197.5 101.646V85.354"/><path id="Line-7" stroke="#D0021B" stroke-linecap="square" d="M277.5 101.646V85.354"/><path id="Line-6" stroke="#D0021B" stroke-linecap="square" d="M197.5 102.5h80"/><text id="3" fill="#D0021B" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="182" y="101">3</tspan></text><text id="class="my"" fill="#417505" font-family="PTMono-Regular, PT Mono" font-size="20" font-weight="normal"><tspan x="185" y="121">class="my"</tspan></text></g></g></svg> \ No newline at end of file diff --git a/9-regular-expressions/12-regexp-anchors/2-test-mac/solution.md b/9-regular-expressions/12-regexp-anchors/2-test-mac/solution.md deleted file mode 100644 index 422bc65e4..000000000 --- a/9-regular-expressions/12-regexp-anchors/2-test-mac/solution.md +++ /dev/null @@ -1,21 +0,0 @@ -A two-digit hex number is `pattern:[0-9a-f]{2}` (assuming the `pattern:i` flag is enabled). - -We need that number `NN`, and then `:NN` repeated 5 times (more numbers); - -The regexp is: `pattern:[0-9a-f]{2}(:[0-9a-f]{2}){5}` - -Now let's show that the match should capture all the text: start at the beginning and end at the end. That's done by wrapping the pattern in `pattern:^...$`. - -Finally: - -```js run -let reg = /^[0-9a-fA-F]{2}(:[0-9a-fA-F]{2}){5}$/i; - -alert( reg.test('01:32:54:67:89:AB') ); // true - -alert( reg.test('0132546789AB') ); // false (no colons) - -alert( reg.test('01:32:54:67:89') ); // false (5 numbers, need 6) - -alert( reg.test('01:32:54:67:89:ZZ') ) // false (ZZ in the end) -``` diff --git a/9-regular-expressions/12-regexp-anchors/article.md b/9-regular-expressions/12-regexp-anchors/article.md deleted file mode 100644 index 0c2dd578a..000000000 --- a/9-regular-expressions/12-regexp-anchors/article.md +++ /dev/null @@ -1,55 +0,0 @@ -# String start ^ and finish $ - -The caret `pattern:'^'` and dollar `pattern:'$'` characters have special meaning in a regexp. They are called "anchors". - -The caret `pattern:^` matches at the beginning of the text, and the dollar `pattern:$` -- in the end. - -For instance, let's test if the text starts with `Mary`: - -```js run -let str1 = "Mary had a little lamb, it's fleece was white as snow"; -let str2 = 'Everywhere Mary went, the lamp was sure to go'; - -alert( /^Mary/.test(str1) ); // true -alert( /^Mary/.test(str2) ); // false -``` - -The pattern `pattern:^Mary` means: "the string start and then Mary". - -Now let's test whether the text ends with an email. - -To match an email, we can use a regexp `pattern:[-.\w]+@([\w-]+\.)+[\w-]{2,20}`. - -To test whether the string ends with the email, let's add `pattern:$` to the pattern: - -```js run -let reg = /[-.\w]+@([\w-]+\.)+[\w-]{2,20}$/g; - -let str1 = 'My email is mail@site.com'; -let str2 = 'Everywhere Mary went, the lamp was sure to go'; - -alert( reg.test(str1) ); // true -alert( reg.test(str2) ); // false -``` - -We can use both anchors together to check whether the string exactly follows the pattern. That's often used for validation. - -For instance we want to check that `str` is exactly a color in the form `#` plus 6 hex digits. The pattern for the color is `pattern:#[0-9a-f]{6}`. - -To check that the *whole string* exactly matches it, we add `pattern:^...$`: - -```js run -let str = "#abcdef"; - -alert( /^#[0-9a-f]{6}$/i.test(str) ); // true -``` - -The regexp engine looks for the text start, then the color, and then immediately the text end. Just what we need. - -```smart header="Anchors have zero length" -Anchors just like `\b` are tests. They have zero-width. - -In other words, they do not match a character, but rather force the regexp engine to check the condition (text start/end). -``` - -The behavior of anchors changes if there's a flag `pattern:m` (multiline mode). We'll explore it in the next chapter. diff --git a/9-regular-expressions/12-regexp-backreferences/article.md b/9-regular-expressions/12-regexp-backreferences/article.md new file mode 100644 index 000000000..b80fa85cf --- /dev/null +++ b/9-regular-expressions/12-regexp-backreferences/article.md @@ -0,0 +1,72 @@ +# Backreferences in pattern: \N and \k<name> + +We can use the contents of capturing groups `pattern:(...)` not only in the result or in the replacement string, but also in the pattern itself. + +## Backreference by number: \N + +A group can be referenced in the pattern using `pattern:\N`, where `N` is the group number. + +To make clear why that's helpful, let's consider a task. + +We need to find quoted strings: either single-quoted `subject:'...'` or a double-quoted `subject:"..."` -- both variants should match. + +How to find them? + +We can put both kinds of quotes in the square brackets: `pattern:['"](.*?)['"]`, but it would find strings with mixed quotes, like `match:"...'` and `match:'..."`. That would lead to incorrect matches when one quote appears inside other ones, like in the string `subject:"She's the one!"`: + +```js run +let str = `He said: "She's the one!".`; + +let regexp = /['"](.*?)['"]/g; + +// The result is not what we'd like to have +alert( str.match(regexp) ); // "She' +``` + +As we can see, the pattern found an opening quote `match:"`, then the text is consumed till the other quote `match:'`, that closes the match. + +To make sure that the pattern looks for the closing quote exactly the same as the opening one, we can wrap it into a capturing group and backreference it: `pattern:(['"])(.*?)\1`. + +Here's the correct code: + +```js run +let str = `He said: "She's the one!".`; + +*!* +let regexp = /(['"])(.*?)\1/g; +*/!* + +alert( str.match(regexp) ); // "She's the one!" +``` + +Now it works! The regular expression engine finds the first quote `pattern:(['"])` and memorizes its content. That's the first capturing group. + +Further in the pattern `pattern:\1` means "find the same text as in the first group", exactly the same quote in our case. + +Similar to that, `pattern:\2` would mean the contents of the second group, `pattern:\3` - the 3rd group, and so on. + +```smart +If we use `?:` in the group, then we can't reference it. Groups that are excluded from capturing `(?:...)` are not memorized by the engine. +``` + +```warn header="Don't mess up: in the pattern `pattern:\1`, in the replacement: `pattern:$1`" +In the replacement string we use a dollar sign: `pattern:$1`, while in the pattern - a backslash `pattern:\1`. +``` + +## Backreference by name: `\k<name>` + +If a regexp has many parentheses, it's convenient to give them names. + +To reference a named group we can use `pattern:\k<name>`. + +In the example below the group with quotes is named `pattern:?<quote>`, so the backreference is `pattern:\k<quote>`: + +```js run +let str = `He said: "She's the one!".`; + +*!* +let regexp = /(?<quote>['"])(.*?)\k<quote>/g; +*/!* + +alert( str.match(regexp) ); // "She's the one!" +``` diff --git a/9-regular-expressions/11-regexp-alternation/01-find-programming-language/solution.md b/9-regular-expressions/13-regexp-alternation/01-find-programming-language/solution.md similarity index 79% rename from 9-regular-expressions/11-regexp-alternation/01-find-programming-language/solution.md rename to 9-regular-expressions/13-regexp-alternation/01-find-programming-language/solution.md index 3419aa498..e33f9cf2f 100644 --- a/9-regular-expressions/11-regexp-alternation/01-find-programming-language/solution.md +++ b/9-regular-expressions/13-regexp-alternation/01-find-programming-language/solution.md @@ -4,11 +4,11 @@ The first idea can be to list the languages with `|` in-between. But that doesn't work right: ```js run -let reg = /Java|JavaScript|PHP|C|C\+\+/g; +let regexp = /Java|JavaScript|PHP|C|C\+\+/g; let str = "Java, JavaScript, PHP, C, C++"; -alert( str.match(reg) ); // Java,Java,PHP,C,C +alert( str.match(regexp) ); // Java,Java,PHP,C,C ``` The regular expression engine looks for alternations one-by-one. That is: first it checks if we have `match:Java`, otherwise -- looks for `match:JavaScript` and so on. @@ -25,9 +25,9 @@ There are two solutions for that problem: In action: ```js run -let reg = /Java(Script)?|C(\+\+)?|PHP/g; +let regexp = /Java(Script)?|C(\+\+)?|PHP/g; let str = "Java, JavaScript, PHP, C, C++"; -alert( str.match(reg) ); // Java,JavaScript,PHP,C,C++ +alert( str.match(regexp) ); // Java,JavaScript,PHP,C,C++ ``` diff --git a/9-regular-expressions/11-regexp-alternation/01-find-programming-language/task.md b/9-regular-expressions/13-regexp-alternation/01-find-programming-language/task.md similarity index 65% rename from 9-regular-expressions/11-regexp-alternation/01-find-programming-language/task.md rename to 9-regular-expressions/13-regexp-alternation/01-find-programming-language/task.md index 61b9526f7..e0f7af95c 100644 --- a/9-regular-expressions/11-regexp-alternation/01-find-programming-language/task.md +++ b/9-regular-expressions/13-regexp-alternation/01-find-programming-language/task.md @@ -5,7 +5,7 @@ There are many programming languages, for instance Java, JavaScript, PHP, C, C++ Create a regexp that finds them in the string `subject:Java JavaScript PHP C++ C`: ```js -let reg = /your regexp/g; +let regexp = /your regexp/g; -alert("Java JavaScript PHP C++ C".match(reg)); // Java JavaScript PHP C++ C +alert("Java JavaScript PHP C++ C".match(regexp)); // Java JavaScript PHP C++ C ``` diff --git a/9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/solution.md b/9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/solution.md new file mode 100644 index 000000000..fb6682b6b --- /dev/null +++ b/9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/solution.md @@ -0,0 +1,23 @@ + +Opening tag is `pattern:\[(b|url|quote)]`. + +Then to find everything till the closing tag -- let's use the pattern `pattern:.*?` with flag `pattern:s` to match any character including the newline and then add a backreference to the closing tag. + +The full pattern: `pattern:\[(b|url|quote)\].*?\[/\1]`. + +In action: + +```js run +let regexp = /\[(b|url|quote)].*?\[\/\1]/gs; + +let str = ` + [b]hello![/b] + [quote] + [url]http://google.com[/url] + [/quote] +`; + +alert( str.match(regexp) ); // [b]hello![/b],[quote][url]http://google.com[/url][/quote] +``` + +Please note that besides escaping `pattern:[`, we had to escape a slash for the closing tag `pattern:[\/\1]`, because normally the slash closes the pattern. diff --git a/9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/task.md b/9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/task.md similarity index 78% rename from 9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/task.md rename to 9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/task.md index e0919e034..72d715afd 100644 --- a/9-regular-expressions/11-regexp-alternation/02-find-matching-bbtags/task.md +++ b/9-regular-expressions/13-regexp-alternation/02-find-matching-bbtags/task.md @@ -15,7 +15,7 @@ Normal: [url] [b]http://google.com[/b] [/url] [quote] [b]text[/b] [/quote] -Impossible: +Can't happen: [b][b]text[/b][/b] ``` @@ -32,17 +32,17 @@ Create a regexp to find all BB-tags with their contents. For instance: ```js -let reg = /your regexp/g; +let regexp = /your regexp/flags; let str = "..[url]http://google.com[/url].."; -alert( str.match(reg) ); // [url]http://google.com[/url] +alert( str.match(regexp) ); // [url]http://google.com[/url] ``` If tags are nested, then we need the outer tag (if we want we can continue the search in its content): ```js -let reg = /your regexp/g; +let regexp = /your regexp/flags; let str = "..[url][b]http://google.com[/b][/url].."; -alert( str.match(reg) ); // [url][b]http://google.com[/b][/url] +alert( str.match(regexp) ); // [url][b]http://google.com[/b][/url] ``` diff --git a/9-regular-expressions/11-regexp-alternation/03-match-quoted-string/solution.md b/9-regular-expressions/13-regexp-alternation/03-match-quoted-string/solution.md similarity index 60% rename from 9-regular-expressions/11-regexp-alternation/03-match-quoted-string/solution.md rename to 9-regular-expressions/13-regexp-alternation/03-match-quoted-string/solution.md index 143be870c..2749f3ff0 100644 --- a/9-regular-expressions/11-regexp-alternation/03-match-quoted-string/solution.md +++ b/9-regular-expressions/13-regexp-alternation/03-match-quoted-string/solution.md @@ -3,15 +3,15 @@ The solution: `pattern:/"(\\.|[^"\\])*"/g`. Step by step: - First we look for an opening quote `pattern:"` -- Then if we have a backslash `pattern:\\` (we technically have to double it in the pattern, because it is a special character, so that's a single backslash in fact), then any character is fine after it (a dot). +- Then if we have a backslash `pattern:\\` (we have to double it in the pattern because it is a special character), then any character is fine after it (a dot). - Otherwise we take any character except a quote (that would mean the end of the string) and a backslash (to prevent lonely backslashes, the backslash is only used with some other symbol after it): `pattern:[^"\\]` - ...And so on till the closing quote. In action: ```js run -let reg = /"(\\.|[^"\\])*"/g; +let regexp = /"(\\.|[^"\\])*"/g; let str = ' .. "test me" .. "Say \\"Hello\\"!" .. "\\\\ \\"" .. '; -alert( str.match(reg) ); // "test me","Say \"Hello\"!","\\ \"" +alert( str.match(regexp) ); // "test me","Say \"Hello\"!","\\ \"" ``` diff --git a/9-regular-expressions/13-regexp-alternation/03-match-quoted-string/task.md b/9-regular-expressions/13-regexp-alternation/03-match-quoted-string/task.md new file mode 100644 index 000000000..27b6bc5c8 --- /dev/null +++ b/9-regular-expressions/13-regexp-alternation/03-match-quoted-string/task.md @@ -0,0 +1,32 @@ +# Find quoted strings + +Create a regexp to find strings in double quotes `subject:"..."`. + +The strings should support escaping, the same way as JavaScript strings do. For instance, quotes can be inserted as `subject:\"` a newline as `subject:\n`, and the backslash itself as `subject:\\`. + +```js +let str = "Just like \"here\"."; +``` + +Please note, in particular, that an escaped quote `subject:\"` does not end a string. + +So we should search from one quote to the other ignoring escaped quotes on the way. + +That's the essential part of the task, otherwise it would be trivial. + +Examples of strings to match: +```js +.. *!*"test me"*/!* .. +.. *!*"Say \"Hello\"!"*/!* ... (escaped quotes inside) +.. *!*"\\"*/!* .. (double backslash inside) +.. *!*"\\ \""*/!* .. (double backslash and an escaped quote inside) +``` + +In JavaScript we need to double the backslashes to pass them right into the string, like this: + +```js run +let str = ' .. "test me" .. "Say \\"Hello\\"!" .. "\\\\ \\"" .. '; + +// the in-memory string +alert(str); // .. "test me" .. "Say \"Hello\"!" .. "\\ \"" .. +``` diff --git a/9-regular-expressions/11-regexp-alternation/04-match-exact-tag/solution.md b/9-regular-expressions/13-regexp-alternation/04-match-exact-tag/solution.md similarity index 72% rename from 9-regular-expressions/11-regexp-alternation/04-match-exact-tag/solution.md rename to 9-regular-expressions/13-regexp-alternation/04-match-exact-tag/solution.md index 70c4de91a..5d4ba8d96 100644 --- a/9-regular-expressions/11-regexp-alternation/04-match-exact-tag/solution.md +++ b/9-regular-expressions/13-regexp-alternation/04-match-exact-tag/solution.md @@ -10,7 +10,7 @@ In the regexp language: `pattern:<style(>|\s.*?>)`. In action: ```js run -let reg = /<style(>|\s.*?>)/g; +let regexp = /<style(>|\s.*?>)/g; -alert( '<style> <styler> <style test="...">'.match(reg) ); // <style>, <style test="..."> +alert( '<style> <styler> <style test="...">'.match(regexp) ); // <style>, <style test="..."> ``` diff --git a/9-regular-expressions/11-regexp-alternation/04-match-exact-tag/task.md b/9-regular-expressions/13-regexp-alternation/04-match-exact-tag/task.md similarity index 68% rename from 9-regular-expressions/11-regexp-alternation/04-match-exact-tag/task.md rename to 9-regular-expressions/13-regexp-alternation/04-match-exact-tag/task.md index 85976a9e7..e8a9e31b4 100644 --- a/9-regular-expressions/11-regexp-alternation/04-match-exact-tag/task.md +++ b/9-regular-expressions/13-regexp-alternation/04-match-exact-tag/task.md @@ -7,7 +7,7 @@ Write a regexp to find the tag `<style...>`. It should match the full tag: it ma For instance: ```js -let reg = /your regexp/g; +let regexp = /your regexp/g; -alert( '<style> <styler> <style test="...">'.match(reg) ); // <style>, <style test="..."> +alert( '<style> <styler> <style test="...">'.match(regexp) ); // <style>, <style test="..."> ``` diff --git a/9-regular-expressions/13-regexp-alternation/article.md b/9-regular-expressions/13-regexp-alternation/article.md new file mode 100644 index 000000000..0fe2175c7 --- /dev/null +++ b/9-regular-expressions/13-regexp-alternation/article.md @@ -0,0 +1,70 @@ +# Alternation (OR) | + +Alternation is the term in regular expression that is actually a simple "OR". + +In a regular expression it is denoted with a vertical line character `pattern:|`. + +For instance, we need to find programming languages: HTML, PHP, Java or JavaScript. + +The corresponding regexp: `pattern:html|php|java(script)?`. + +A usage example: + +```js run +let regexp = /html|php|css|java(script)?/gi; + +let str = "First HTML appeared, then CSS, then JavaScript"; + +alert( str.match(regexp) ); // 'HTML', 'CSS', 'JavaScript' +``` + +We already saw a similar thing -- square brackets. They allow to choose between multiple characters, for instance `pattern:gr[ae]y` matches `match:gray` or `match:grey`. + +Square brackets allow only characters or character classes. Alternation allows any expressions. A regexp `pattern:A|B|C` means one of expressions `A`, `B` or `C`. + +For instance: + +- `pattern:gr(a|e)y` means exactly the same as `pattern:gr[ae]y`. +- `pattern:gra|ey` means `match:gra` or `match:ey`. + +To apply alternation to a chosen part of the pattern, we can enclose it in parentheses: +- `pattern:I love HTML|CSS` matches `match:I love HTML` or `match:CSS`. +- `pattern:I love (HTML|CSS)` matches `match:I love HTML` or `match:I love CSS`. + +## Example: regexp for time + +In previous articles there was a task to build a regexp for searching time in the form `hh:mm`, for instance `12:00`. But a simple `pattern:\d\d:\d\d` is too vague. It accepts `25:99` as the time (as 99 minutes match the pattern, but that time is invalid). + +How can we make a better pattern? + +We can use more careful matching. First, the hours: + +- If the first digit is `0` or `1`, then the next digit can be any: `pattern:[01]\d`. +- Otherwise, if the first digit is `2`, then the next must be `pattern:[0-3]`. +- (no other first digit is allowed) + +We can write both variants in a regexp using alternation: `pattern:[01]\d|2[0-3]`. + +Next, minutes must be from `00` to `59`. In the regular expression language that can be written as `pattern:[0-5]\d`: the first digit `0-5`, and then any digit. + +If we glue hours and minutes together, we get the pattern: `pattern:[01]\d|2[0-3]:[0-5]\d`. + +We're almost done, but there's a problem. The alternation `pattern:|` now happens to be between `pattern:[01]\d` and `pattern:2[0-3]:[0-5]\d`. + +That is: minutes are added to the second alternation variant, here's a clear picture: + +``` +[01]\d | 2[0-3]:[0-5]\d +``` + +That pattern looks for `pattern:[01]\d` or `pattern:2[0-3]:[0-5]\d`. + +But that's wrong, the alternation should only be used in the "hours" part of the regular expression, to allow `pattern:[01]\d` OR `pattern:2[0-3]`. Let's correct that by enclosing "hours" into parentheses: `pattern:([01]\d|2[0-3]):[0-5]\d`. + +The final solution: + +```js run +let regexp = /([01]\d|2[0-3]):[0-5]\d/g; + +alert("00:00 10:10 23:59 25:99 1:2".match(regexp)); // 00:00,10:10,23:59 +``` diff --git a/9-regular-expressions/13-regexp-multiline-mode/article.md b/9-regular-expressions/13-regexp-multiline-mode/article.md deleted file mode 100644 index 1b5623dc8..000000000 --- a/9-regular-expressions/13-regexp-multiline-mode/article.md +++ /dev/null @@ -1,76 +0,0 @@ -# Multiline mode, flag "m" - -The multiline mode is enabled by the flag `pattern:/.../m`. - -It only affects the behavior of `pattern:^` and `pattern:$`. - -In the multiline mode they match not only at the beginning and end of the string, but also at start/end of line. - -## Line start ^ - -In the example below the text has multiple lines. The pattern `pattern:/^\d+/gm` takes a number from the beginning of each one: - -```js run -let str = `1st place: Winnie -2nd place: Piglet -33rd place: Eeyore`; - -*!* -alert( str.match(/^\d+/gm) ); // 1, 2, 33 -*/!* -``` - -Without the flag `pattern:/.../m` only the first number is matched: - - -```js run -let str = `1st place: Winnie -2nd place: Piglet -33rd place: Eeyore`; - -*!* -alert( str.match(/^\d+/g) ); // 1 -*/!* -``` - -That's because by default a caret `pattern:^` only matches at the beginning of the text, and in the multiline mode -- at the start of a line. - -The regular expression engine moves along the text and looks for a string start `pattern:^`, when finds -- continues to match the rest of the pattern `pattern:\d+`. - -## Line end $ - -The dollar sign `pattern:$` behaves similarly. - -The regular expression `pattern:\w+$` finds the last word in every line - -```js run -let str = `1st place: Winnie -2nd place: Piglet -33rd place: Eeyore`; - -alert( str.match(/\w+$/gim) ); // Winnie,Piglet,Eeyore -``` - -Without the `pattern:/.../m` flag the dollar `pattern:$` would only match the end of the whole string, so only the very last word would be found. - -## Anchors ^$ versus \n - -To find a newline, we can use not only `pattern:^` and `pattern:$`, but also the newline character `\n`. - -The first difference is that unlike anchors, the character `\n` "consumes" the newline character and adds it to the result. - -For instance, here we use it instead of `pattern:$`: - -```js run -let str = `1st place: Winnie -2nd place: Piglet -33rd place: Eeyore`; - -alert( str.match(/\w+\n/gim) ); // Winnie\n,Piglet\n -``` - -Here every match is a word plus a newline character. - -And one more difference -- the newline `\n` does not match at the string end. That's why `Eeyore` is not found in the example above. - -So, anchors are usually better, they are closer to what we want to get. diff --git a/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/solution.md b/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/solution.md new file mode 100644 index 000000000..ebc12689d --- /dev/null +++ b/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/solution.md @@ -0,0 +1,28 @@ + +The regexp for an integer number is `pattern:\d+`. + +We can exclude negatives by prepending it with the negative lookbehind: `pattern:(?<!-)\d+`. + +Although, if we try it now, we may notice one more "extra" result: + +```js run +let regexp = /(?<!-)\d+/g; + +let str = "0 12 -5 123 -18"; + +console.log( str.match(regexp) ); // 0, 12, 123, *!*8*/!* +``` + +As you can see, it matches `match:8`, from `subject:-18`. To exclude it, we need to ensure that the regexp starts matching a number not from the middle of another (non-matching) number. + +We can do it by specifying another negative lookbehind: `pattern:(?<!-)(?<!\d)\d+`. Now `pattern:(?<!\d)` ensures that a match does not start after another digit, just what we need. + +We can also join them into a single lookbehind here: + +```js run +let regexp = /(?<![-\d])\d+/g; + +let str = "0 12 -5 123 -18"; + +alert( str.match(regexp) ); // 0, 12, 123 +``` diff --git a/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/task.md b/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/task.md new file mode 100644 index 000000000..cd58f9923 --- /dev/null +++ b/9-regular-expressions/14-regexp-lookahead-lookbehind/1-find-non-negative-integers/task.md @@ -0,0 +1,14 @@ +# Find non-negative integers + +There's a string of integer numbers. + +Create a regexp that looks for only non-negative ones (zero is allowed). + +An example of use: +```js +let regexp = /your regexp/g; + +let str = "0 12 -5 123 -18"; + +alert( str.match(regexp) ); // 0, 12, 123 +``` diff --git a/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/solution.md b/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/solution.md new file mode 100644 index 000000000..68bca8842 --- /dev/null +++ b/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/solution.md @@ -0,0 +1,36 @@ +In order to insert after the `<body>` tag, we must first find it. We can use the regular expression pattern `pattern:<body.*?>` for that. + +In this task, we don't need to modify the `<body>` tag. We only need to add the text after it. + +Here's how we can do it: + +```js run +let str = '...<body style="...">...'; +str = str.replace(/<body.*?>/, '$&<h1>Hello</h1>'); + +alert(str); // ...<body style="..."><h1>Hello</h1>... +``` + +In the replacement string `$&` means the match itself, that is, the part of the source text that corresponds to `pattern:<body.*?>`. It gets replaced by itself plus `<h1>Hello</h1>`. + +An alternative is to use lookbehind: + +```js run +let str = '...<body style="...">...'; +str = str.replace(/(?<=<body.*?>)/, `<h1>Hello</h1>`); + +alert(str); // ...<body style="..."><h1>Hello</h1>... +``` + +As you can see, there's only lookbehind part in this regexp. + +It works like this: +- At every position in the text. +- Check if it's preceded by `pattern:<body.*?>`. +- If it's so, then we have the match. + +The tag `pattern:<body.*?>` won't be returned. The result of this regexp is literally an empty string, but it matches only at positions preceded by `pattern:<body.*?>`. + +So it replaces the "empty line", preceded by `pattern:<body.*?>`, with `<h1>Hello</h1>`. That's the insertion after `<body>`. + +P.S. Regexp flags, such as `pattern:s` and `pattern:i` can also be useful: `pattern:/<body.*?>/si`. The `pattern:s` flag makes the dot `pattern:.` match a newline character, and `pattern:i` flag makes `pattern:<body>` also match `match:<BODY>` case-insensitively. diff --git a/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/task.md b/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/task.md new file mode 100644 index 000000000..5ee42798a --- /dev/null +++ b/9-regular-expressions/14-regexp-lookahead-lookbehind/2-insert-after-head/task.md @@ -0,0 +1,31 @@ +# Insert After Head + +We have a string with an HTML Document. + +Write a regular expression that inserts `<h1>Hello</h1>` immediately after `<body>` tag. The tag may have attributes. + +For instance: + +```js +let regexp = /your regular expression/; + +let str = ` +<html> + <body style="height: 200px"> + ... + </body> +</html> +`; + +str = str.replace(regexp, `<h1>Hello</h1>`); +``` + +After that the value of `str` should be: + +```html +<html> + <body style="height: 200px"><h1>Hello</h1> + ... + </body> +</html> +``` diff --git a/9-regular-expressions/14-regexp-lookahead-lookbehind/article.md b/9-regular-expressions/14-regexp-lookahead-lookbehind/article.md index 376d52d66..cee8215f9 100644 --- a/9-regular-expressions/14-regexp-lookahead-lookbehind/article.md +++ b/9-regular-expressions/14-regexp-lookahead-lookbehind/article.md @@ -1,105 +1,134 @@ # Lookahead and lookbehind -Sometimes we need to match a pattern only if followed by another pattern. For instance, we'd like to get the price from a string like `subject:1 turkey costs 30€`. +Sometimes we need to find only those matches for a pattern that are followed or preceded by another pattern. -We need a number (let's say a price has no decimal point) followed by `subject:€` sign. +There's a special syntax for that, called "lookahead" and "lookbehind", together referred to as "lookaround". -That's what lookahead is for. +For the start, let's find the price from the string like `subject:1 turkey costs 30€`. That is: a number, followed by `subject:€` sign. ## Lookahead -The syntax is: `pattern:x(?=y)`, it means "look for `pattern:x`, but match only if followed by `pattern:y`". +The syntax is: `pattern:X(?=Y)`, it means "look for `pattern:X`, but match only if followed by `pattern:Y`". There may be any pattern instead of `pattern:X` and `pattern:Y`. -For an integer amount followed by `subject:€`, the regexp will be `pattern:\d+(?=€)`: +For an integer number followed by `subject:€`, the regexp will be `pattern:\d+(?=€)`: ```js run let str = "1 turkey costs 30€"; -alert( str.match(/\d+(?=€)/) ); // 30 (correctly skipped the sole number 1) +alert( str.match(/\d+(?=€)/) ); // 30, the number 1 is ignored, as it's not followed by € ``` -Let's say we want a quantity instead, that is a number, NOT followed by `subject:€`. +Please note: the lookahead is merely a test, the contents of the parentheses `pattern:(?=...)` is not included in the result `match:30`. -Here a negative lookahead can be applied. +When we look for `pattern:X(?=Y)`, the regular expression engine finds `pattern:X` and then checks if there's `pattern:Y` immediately after it. If it's not so, then the potential match is skipped, and the search continues. -The syntax is: `pattern:x(?!y)`, it means "search `pattern:x`, but only if not followed by `pattern:y`". +More complex tests are possible, e.g. `pattern:X(?=Y)(?=Z)` means: + +1. Find `pattern:X`. +2. Check if `pattern:Y` is immediately after `pattern:X` (skip if isn't). +3. Check if `pattern:Z` is also immediately after `pattern:X` (skip if isn't). +4. If both tests passed, then the `pattern:X` is a match, otherwise continue searching. + +In other words, such pattern means that we're looking for `pattern:X` followed by `pattern:Y` and `pattern:Z` at the same time. + +That's only possible if patterns `pattern:Y` and `pattern:Z` aren't mutually exclusive. + +For example, `pattern:\d+(?=\s)(?=.*30)` looks for `pattern:\d+` that is followed by a space `pattern:(?=\s)`, and there's `30` somewhere after it `pattern:(?=.*30)`: + +```js run +let str = "1 turkey costs 30€"; + +alert( str.match(/\d+(?=\s)(?=.*30)/) ); // 1 +``` + +In our string that exactly matches the number `1`. + +## Negative lookahead + +Let's say that we want a quantity instead, not a price from the same string. That's a number `pattern:\d+`, NOT followed by `subject:€`. + +For that, a negative lookahead can be applied. + +The syntax is: `pattern:X(?!Y)`, it means "search `pattern:X`, but only if not followed by `pattern:Y`". ```js run let str = "2 turkeys cost 60€"; -alert( str.match(/\d+(?!€)/) ); // 2 (correctly skipped the price) +alert( str.match(/\d+\b(?!€)/g) ); // 2 (the price is not matched) ``` ## Lookbehind -Lookahead allows to add a condition for "what goes after". +```warn header="Lookbehind browser compatibility" +Please Note: Lookbehind is not supported in non-V8 browsers, such as Safari, Internet Explorer. +``` + +Lookahead allows to add a condition for "what follows". -Lookbehind is similar, but it looks behind. That is, it allows to match a pattern only if there's something before. +Lookbehind is similar, but it looks behind. That is, it allows to match a pattern only if there's something before it. The syntax is: -- Positive lookbehind: `pattern:(?<=y)x`, matches `pattern:x`, but only if it follows after `pattern:y`. -- Negative lookbehind: `pattern:(?<!y)x`, matches `pattern:x`, but only if there's no `pattern:y` before. +- Positive lookbehind: `pattern:(?<=Y)X`, matches `pattern:X`, but only if there's `pattern:Y` before it. +- Negative lookbehind: `pattern:(?<!Y)X`, matches `pattern:X`, but only if there's no `pattern:Y` before it. -For example, let's change the price to US dollars. The dollar sign is usually before the number, so to look for `$30` we'll use `pattern:(?<=\$)\d+` -- an amount preceeded by `subject:$`: +For example, let's change the price to US dollars. The dollar sign is usually before the number, so to look for `$30` we'll use `pattern:(?<=\$)\d+` -- an amount preceded by `subject:$`: ```js run let str = "1 turkey costs $30"; +// the dollar sign is escaped \$ alert( str.match(/(?<=\$)\d+/) ); // 30 (skipped the sole number) ``` -And, to find the quantity -- a number, not preceeded by `subject:$`, we can use a negative lookbehind `pattern:(?<!\$)\d+`: +And, if we need the quantity -- a number, not preceded by `subject:$`, then we can use a negative lookbehind `pattern:(?<!\$)\d+`: ```js run let str = "2 turkeys cost $60"; -alert( str.match(/(?<!\$)\d+/) ); // 2 (skipped the price) +alert( str.match(/(?<!\$)\b\d+/g) ); // 2 (the price is not matched) ``` -## Capture groups +## Capturing groups -Generally, what's inside the lookaround (a common name for both lookahead and lookbehind) parentheses does not become a part of the match. +Generally, the contents inside lookaround parentheses does not become a part of the result. -E.g. in the pattern `pattern:\d+(?!€)`, the `pattern:€` sign doesn't get captured as a part of the match. +E.g. in the pattern `pattern:\d+(?=€)`, the `pattern:€` sign doesn't get captured as a part of the match. That's natural: we look for a number `pattern:\d+`, while `pattern:(?=€)` is just a test that it should be followed by `subject:€`. -But if we want to capture the whole lookaround expression or a part of it, that's possible. Just need to wrap that into additional parentheses. +But in some situations we might want to capture the lookaround expression as well, or a part of it. That's possible. Just wrap that part into additional parentheses. -For instance, here the currency `pattern:(€|kr)` is captured, along with the amount: +In the example below the currency sign `pattern:(€|kr)` is captured, along with the amount: ```js run let str = "1 turkey costs 30€"; -let reg = /\d+(?=(€|kr))/; // extra parentheses around €|kr +let regexp = /\d+(?=(€|kr))/; // extra parentheses around €|kr -alert( str.match(reg) ); // 30, € +alert( str.match(regexp) ); // 30, € ``` And here's the same for lookbehind: ```js run let str = "1 turkey costs $30"; -let reg = /(?<=(\$|£))\d+/; +let regexp = /(?<=(\$|£))\d+/; -alert( str.match(reg) ); // 30, $ +alert( str.match(regexp) ); // 30, $ ``` -Please note that for lookbehind the order stays be same, even though lookahead parentheses are before the main pattern. +## Summary -Usually parentheses are numbered left-to-right, but lookbehind is an exception, it is always captured after the main pattern. So the match for `pattern:\d+` goes in the result first, and then for `pattern:(\$|£)`. +Lookahead and lookbehind (commonly referred to as "lookaround") are useful when we'd like to match something depending on the context before/after it. +For simple regexps we can do the similar thing manually. That is: match everything, in any context, and then filter by context in the loop. -## Summary +Remember, `str.match` (without flag `pattern:g`) and `str.matchAll` (always) return matches as arrays with `index` property, so we know where exactly in the text it is, and can check the context. -Lookahead and lookbehind (commonly referred to as "lookaround") are useful for simple regular expressions, when we'd like not to take something into the match depending on the context before/after it. - -Sometimes we can do the same manually, that is: match all and filter by context in the loop. Remember, `str.matchAll` and `reg.exec` return matches with `.index` property, so we know where exactly in the text it is. But generally regular expressions can do it better. +But generally lookaround is more convenient. Lookaround types: | Pattern | type | matches | |--------------------|------------------|---------| -| `pattern:x(?=y)` | Positive lookahead | `x` if followed by `y` | -| `pattern:x(?!y)` | Negative lookahead | `x` if not followed by `y` | -| `pattern:(?<=y)x` | Positive lookbehind | `x` if after `y` | -| `pattern:(?<!y)x` | Negative lookbehind | `x` if not after `y` | - -Lookahead can also used to disable backtracking. Why that may be needed -- see in the next chapter. +| `X(?=Y)` | Positive lookahead | `pattern:X` if followed by `pattern:Y` | +| `X(?!Y)` | Negative lookahead | `pattern:X` if not followed by `pattern:Y` | +| `(?<=Y)X` | Positive lookbehind | `pattern:X` if after `pattern:Y` | +| `(?<!Y)X` | Negative lookbehind | `pattern:X` if not after `pattern:Y` | diff --git a/9-regular-expressions/15-regexp-catastrophic-backtracking/article.md b/9-regular-expressions/15-regexp-catastrophic-backtracking/article.md new file mode 100644 index 000000000..c1d4040f7 --- /dev/null +++ b/9-regular-expressions/15-regexp-catastrophic-backtracking/article.md @@ -0,0 +1,318 @@ +# Catastrophic backtracking + +Some regular expressions are looking simple, but can execute a veeeeeery long time, and even "hang" the JavaScript engine. + +Sooner or later most developers occasionally face such behavior. The typical symptom -- a regular expression works fine sometimes, but for certain strings it "hangs", consuming 100% of CPU. + +In such case a web-browser suggests to kill the script and reload the page. Not a good thing for sure. + +For server-side JavaScript such a regexp may hang the server process, that's even worse. So we definitely should take a look at it. + +## Example + +Let's say we have a string, and we'd like to check if it consists of words `pattern:\w+` with an optional space `pattern:\s?` after each. + +An obvious way to construct a regexp would be to take a word followed by an optional space `pattern:\w+\s?` and then repeat it with `*`. + +That leads us to the regexp `pattern:^(\w+\s?)*$`, it specifies zero or more such words, that start at the beginning `pattern:^` and finish at the end `pattern:$` of the line. + +In action: + +```js run +let regexp = /^(\w+\s?)*$/; + +alert( regexp.test("A good string") ); // true +alert( regexp.test("Bad characters: $@#") ); // false +``` + +The regexp seems to work. The result is correct. Although, on certain strings it takes a lot of time. So long that JavaScript engine "hangs" with 100% CPU consumption. + +If you run the example below, you probably won't see anything, as JavaScript will just "hang". A web-browser will stop reacting on events, the UI will stop working (most browsers allow only scrolling). After some time it will suggest to reload the page. So be careful with this: + +```js run +let regexp = /^(\w+\s?)*$/; +let str = "An input string that takes a long time or even makes this regexp hang!"; + +// will take a very long time +alert( regexp.test(str) ); +``` + +To be fair, let's note that some regular expression engines can handle such a search effectively, for example V8 engine version starting from 8.8 can do that (so Google Chrome 88 doesn't hang here), while Firefox browser does hang. + +## Simplified example + +What's the matter? Why does the regular expression hang? + +To understand that, let's simplify the example: remove spaces `pattern:\s?`. Then it becomes `pattern:^(\w+)*$`. + +And, to make things more obvious, let's replace `pattern:\w` with `pattern:\d`. The resulting regular expression still hangs, for instance: + +```js run +let regexp = /^(\d+)*$/; + +let str = "012345678901234567890123456789z"; + +// will take a very long time (careful!) +alert( regexp.test(str) ); +``` + +So what's wrong with the regexp? + +First, one may notice that the regexp `pattern:(\d+)*` is a little bit strange. The quantifier `pattern:*` looks extraneous. If we want a number, we can use `pattern:\d+`. + +Indeed, the regexp is artificial; we got it by simplifying the previous example. But the reason why it is slow is the same. So let's understand it, and then the previous example will become obvious. + +What happens during the search of `pattern:^(\d+)*$` in the line `subject:123456789z` (shortened a bit for clarity, please note a non-digit character `subject:z` at the end, it's important), why does it take so long? + +Here's what the regexp engine does: + +1. First, the regexp engine tries to find the content of the parentheses: the number `pattern:\d+`. The plus `pattern:+` is greedy by default, so it consumes all digits: + + ``` + \d+....... + (123456789)z + ``` + + After all digits are consumed, `pattern:\d+` is considered found (as `match:123456789`). + + Then the star quantifier `pattern:(\d+)*` applies. But there are no more digits in the text, so the star doesn't give anything. + + The next character in the pattern is the string end `pattern:$`. But in the text we have `subject:z` instead, so there's no match: + + ``` + X + \d+........$ + (123456789)z + ``` + +2. As there's no match, the greedy quantifier `pattern:+` decreases the count of repetitions, backtracks one character back. + + Now `pattern:\d+` takes all digits except the last one (`match:12345678`): + ``` + \d+....... + (12345678)9z + ``` +3. Then the engine tries to continue the search from the next position (right after `match:12345678`). + + The star `pattern:(\d+)*` can be applied -- it gives one more match of `pattern:\d+`, the number `match:9`: + + ``` + + \d+.......\d+ + (12345678)(9)z + ``` + + The engine tries to match `pattern:$` again, but fails, because it meets `subject:z` instead: + + ``` + X + \d+.......\d+ + (12345678)(9)z + ``` + + +4. There's no match, so the engine will continue backtracking, decreasing the number of repetitions. Backtracking generally works like this: the last greedy quantifier decreases the number of repetitions until it reaches the minimum. Then the previous greedy quantifier decreases, and so on. + + All possible combinations are attempted. Here are their examples. + + The first number `pattern:\d+` has 7 digits, and then a number of 2 digits: + + ``` + X + \d+......\d+ + (1234567)(89)z + ``` + + The first number has 7 digits, and then two numbers of 1 digit each: + + ``` + X + \d+......\d+\d+ + (1234567)(8)(9)z + ``` + + The first number has 6 digits, and then a number of 3 digits: + + ``` + X + \d+.......\d+ + (123456)(789)z + ``` + + The first number has 6 digits, and then 2 numbers: + + ``` + X + \d+.....\d+ \d+ + (123456)(78)(9)z + ``` + + ...And so on. + + +There are many ways to split a sequence of digits `123456789` into numbers. To be precise, there are <code>2<sup>n</sup>-1</code>, where `n` is the length of the sequence. + +- For `123456789` we have `n=9`, that gives 511 combinations. +- For a longer sequence with `n=20` there are about one million (1048575) combinations. +- For `n=30` - a thousand times more (1073741823 combinations). + +Trying each of them is exactly the reason why the search takes so long. + +## Back to words and strings + +The similar thing happens in our first example, when we look for words by pattern `pattern:^(\w+\s?)*$` in the string `subject:An input that hangs!`. + +The reason is that a word can be represented as one `pattern:\w+` or many: + +``` +(input) +(inpu)(t) +(inp)(u)(t) +(in)(p)(ut) +... +``` + +For a human, it's obvious that there may be no match, because the string ends with an exclamation sign `!`, but the regular expression expects a wordly character `pattern:\w` or a space `pattern:\s` at the end. But the engine doesn't know that. + +It tries all combinations of how the regexp `pattern:(\w+\s?)*` can "consume" the string, including variants with spaces `pattern:(\w+\s)*` and without them `pattern:(\w+)*` (because spaces `pattern:\s?` are optional). As there are many such combinations (we've seen it with digits), the search takes a lot of time. + +What to do? + +Should we turn on the lazy mode? + +Unfortunately, that won't help: if we replace `pattern:\w+` with `pattern:\w+?`, the regexp will still hang. The order of combinations will change, but not their total count. + +Some regular expression engines have tricky tests and finite automations that allow to avoid going through all combinations or make it much faster, but most engines don't, and it doesn't always help. + +## How to fix? + +There are two main approaches to fixing the problem. + +The first is to lower the number of possible combinations. + +Let's make the space non-optional by rewriting the regular expression as `pattern:^(\w+\s)*\w*$` - we'll look for any number of words followed by a space `pattern:(\w+\s)*`, and then (optionally) a final word `pattern:\w*`. + +This regexp is equivalent to the previous one (matches the same) and works well: + +```js run +let regexp = /^(\w+\s)*\w*$/; +let str = "An input string that takes a long time or even makes this regex hang!"; + +alert( regexp.test(str) ); // false +``` + +Why did the problem disappear? + +That's because now the space is mandatory. + +The previous regexp, if we omit the space, becomes `pattern:(\w+)*`, leading to many combinations of `\w+` within a single word + +So `subject:input` could be matched as two repetitions of `pattern:\w+`, like this: + +``` +\w+ \w+ +(inp)(ut) +``` + +The new pattern is different: `pattern:(\w+\s)*` specifies repetitions of words followed by a space! The `subject:input` string can't be matched as two repetitions of `pattern:\w+\s`, because the space is mandatory. + +The time needed to try a lot of (actually most of) combinations is now saved. + +## Preventing backtracking + +It's not always convenient to rewrite a regexp though. In the example above it was easy, but it's not always obvious how to do it. + +Besides, a rewritten regexp is usually more complex, and that's not good. Regexps are complex enough without extra efforts. + +Luckily, there's an alternative approach. We can forbid backtracking for the quantifier. + +The root of the problem is that the regexp engine tries many combinations that are obviously wrong for a human. + +E.g. in the regexp `pattern:(\d+)*$` it's obvious for a human, that `pattern:+` shouldn't backtrack. If we replace one `pattern:\d+` with two separate `pattern:\d+\d+`, nothing changes: + +``` +\d+........ +(123456789)! + +\d+...\d+.... +(1234)(56789)! +``` + +And in the original example `pattern:^(\w+\s?)*$` we may want to forbid backtracking in `pattern:\w+`. That is: `pattern:\w+` should match a whole word, with the maximal possible length. There's no need to lower the repetitions count in `pattern:\w+` or to split it into two words `pattern:\w+\w+` and so on. + +Modern regular expression engines support possessive quantifiers for that. Regular quantifiers become possessive if we add `pattern:+` after them. That is, we use `pattern:\d++` instead of `pattern:\d+` to stop `pattern:+` from backtracking. + +Possessive quantifiers are in fact simpler than "regular" ones. They just match as many as they can, without any backtracking. The search process without backtracking is simpler. + +There are also so-called "atomic capturing groups" - a way to disable backtracking inside parentheses. + +...But the bad news is that, unfortunately, in JavaScript they are not supported. + +We can emulate them though using a "lookahead transform". + +### Lookahead to the rescue! + +So we've come to real advanced topics. We'd like a quantifier, such as `pattern:+` not to backtrack, because sometimes backtracking makes no sense. + +The pattern to take as many repetitions of `pattern:\w` as possible without backtracking is: `pattern:(?=(\w+))\1`. Of course, we could take another pattern instead of `pattern:\w`. + +That may seem odd, but it's actually a very simple transform. + +Let's decipher it: + +- Lookahead `pattern:?=` looks forward for the longest word `pattern:\w+` starting at the current position. +- The contents of parentheses with `pattern:?=...` isn't memorized by the engine, so wrap `pattern:\w+` into parentheses. Then the engine will memorize their contents +- ...And allow us to reference it in the pattern as `pattern:\1`. + +That is: we look ahead - and if there's a word `pattern:\w+`, then match it as `pattern:\1`. + +Why? That's because the lookahead finds a word `pattern:\w+` as a whole and we capture it into the pattern with `pattern:\1`. So we essentially implemented a possessive plus `pattern:+` quantifier. It captures only the whole word `pattern:\w+`, not a part of it. + +For instance, in the word `subject:JavaScript` it may not only match `match:Java`, but leave out `match:Script` to match the rest of the pattern. + +Here's the comparison of two patterns: + +```js run +alert( "JavaScript".match(/\w+Script/)); // JavaScript +alert( "JavaScript".match(/(?=(\w+))\1Script/)); // null +``` + +1. In the first variant `pattern:\w+` first captures the whole word `subject:JavaScript` but then `pattern:+` backtracks character by character, to try to match the rest of the pattern, until it finally succeeds (when `pattern:\w+` matches `match:Java`). +2. In the second variant `pattern:(?=(\w+))` looks ahead and finds the word `subject:JavaScript`, that is included into the pattern as a whole by `pattern:\1`, so there remains no way to find `subject:Script` after it. + +We can put a more complex regular expression into `pattern:(?=(\w+))\1` instead of `pattern:\w`, when we need to forbid backtracking for `pattern:+` after it. + +```smart +There's more about the relation between possessive quantifiers and lookahead in articles [Regex: Emulate Atomic Grouping (and Possessive Quantifiers) with LookAhead](https://instanceof.me/post/52245507631/regex-emulate-atomic-grouping-with-lookahead) and [Mimicking Atomic Groups](https://blog.stevenlevithan.com/archives/mimic-atomic-groups). +``` + +Let's rewrite the first example using lookahead to prevent backtracking: + +```js run +let regexp = /^((?=(\w+))\2\s?)*$/; + +alert( regexp.test("A good string") ); // true + +let str = "An input string that takes a long time or even makes this regex hang!"; + +alert( regexp.test(str) ); // false, works and fast! +``` + +Here `pattern:\2` is used instead of `pattern:\1`, because there are additional outer parentheses. To avoid messing up with the numbers, we can give the parentheses a name, e.g. `pattern:(?<word>\w+)`. + +```js run +// parentheses are named ?<word>, referenced as \k<word> +let regexp = /^((?=(?<word>\w+))\k<word>\s?)*$/; + +let str = "An input string that takes a long time or even makes this regex hang!"; + +alert( regexp.test(str) ); // false + +alert( regexp.test("A correct string") ); // true +``` + +The problem described in this article is called "catastrophic backtracking". + +We covered two ways how to solve it: +- Rewrite the regexp to lower the possible combinations count. +- Prevent backtracking. diff --git a/9-regular-expressions/15-regexp-infinite-backtracking-problem/article.md b/9-regular-expressions/15-regexp-infinite-backtracking-problem/article.md deleted file mode 100644 index 54bbccd9b..000000000 --- a/9-regular-expressions/15-regexp-infinite-backtracking-problem/article.md +++ /dev/null @@ -1,293 +0,0 @@ -# Infinite backtracking problem - -Some regular expressions are looking simple, but can execute veeeeeery long time, and even "hang" the JavaScript engine. - -Sooner or later most developers occasionally face such behavior. - -The typical situation -- a regular expression works fine sometimes, but for certain strings it "hangs" consuming 100% of CPU. - -In a web-browser it kills the page. Not a good thing for sure. - -For server-side JavaScript it may become a vulnerability, and it uses regular expressions to process user data. Bad input will make the process hang, causing denial of service. The author personally saw and reported such vulnerabilities even for very well-known and widely used programs. - -So the problem is definitely worth to deal with. - -## Introduction - -The plan will be like this: - -1. First we see the problem how it may occur. -2. Then we simplify the situation and see why it occurs. -3. Then we fix it. - -For instance let's consider searching tags in HTML. - -We want to find all tags, with or without attributes -- like `subject:<a href="..." class="doc" ...>`. We need the regexp to work reliably, because HTML comes from the internet and can be messy. - -In particular, we need it to match tags like `<a test="<>" href="#">` -- with `<` and `>` in attributes. That's allowed by [HTML standard](https://html.spec.whatwg.org/multipage/syntax.html#syntax-attributes). - -Now we can see that a simple regexp like `pattern:<[^>]+>` doesn't work, because it stops at the first `>`, and we need to ignore `<>` if inside an attribute. - -```js run -// the match doesn't reach the end of the tag - wrong! -alert( '<a test="<>" href="#">'.match(/<[^>]+>/) ); // <a test="<> -``` - -To correctly handle such situations we need a more complex regular expression. It will have the form `pattern:<tag (key=value)*>`. - -1. For the `tag` name: `pattern:\w+`, -2. For the `key` name: `pattern:\w+`, -3. And the `value`: a quoted string `pattern:"[^"]*"`. - -If we substitute these into the pattern above and throw in some optional spaces `pattern:\s`, the full regexp becomes: `pattern:<\w+(\s*\w+="[^"]*"\s*)*>`. - -That regexp is not perfect! It doesn't yet support all details of HTML, for instance unquoted values, and there are other ways to improve, but let's not add complexity. It will demonstrate the problem for us. - -The regexp seems to work: - -```js run -let reg = /<\w+(\s*\w+="[^"]*"\s*)*>/g; - -let str='...<a test="<>" href="#">... <b>...'; - -alert( str.match(reg) ); // <a test="<>" href="#">, <b> -``` - -Great! It found both the long tag `match:<a test="<>" href="#">` and the short one `match:<b>`. - -Now, that we've got a seemingly working solution, let's get to the infinite backtracking itself. - -## Infinite backtracking - -If you run our regexp on the input below, it may hang the browser (or another JavaScript host): - -```js run -let reg = /<\w+(\s*\w+="[^"]*"\s*)*>/g; - -let str = `<tag a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" - a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b" a="b"`; - -*!* -// The search will take a long, long time -alert( str.match(reg) ); -*/!* -``` - -Some regexp engines can handle that search, but most of them can't. - -What's the matter? Why a simple regular expression "hangs" on such a small string? - -Let's simplify the regexp by stripping the tag name and the quotes. So that we look only for `key=value` attributes: `pattern:<(\s*\w+=\w+\s*)*>`. - -Unfortunately, the regexp still hangs: - -```js run -// only search for space-delimited attributes -let reg = /<(\s*\w+=\w+\s*)*>/g; - -let str = `<a=b a=b a=b a=b a=b a=b a=b a=b - a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b`; - -*!* -// the search will take a long, long time -alert( str.match(reg) ); -*/!* -``` - -Here we end the demo of the problem and start looking into what's going on, why it hangs and how to fix it. - -## Detailed example - -To make an example even simpler, let's consider `pattern:(\d+)*$`. - -This regular expression also has the same problem. In most regexp engines that search takes a very long time (careful -- can hang): - -```js run -alert( '12345678901234567890123456789123456789z'.match(/(\d+)*$/) ); -``` - -So what's wrong with the regexp? - -First, one may notice that the regexp is a little bit strange. The quantifier `pattern:*` looks extraneous. If we want a number, we can use `pattern:\d+$`. - -Indeed, the regexp is artificial. But the reason why it is slow is the same as those we saw above. So let's understand it, and then the previous example will become obvious. - -What happen during the search of `pattern:(\d+)*$` in the line `subject:123456789z`? - -1. First, the regexp engine tries to find a number `pattern:\d+`. The plus `pattern:+` is greedy by default, so it consumes all digits: - - ``` - \d+....... - (123456789)z - ``` -2. Then it tries to apply the star quantifier, but there are no more digits, so it the star doesn't give anything. - -3. Then the pattern expects to see the string end `pattern:$`, and in the text we have `subject:z`, so there's no match: - - ``` - X - \d+........$ - (123456789)z - ``` - -4. As there's no match, the greedy quantifier `pattern:+` decreases the count of repetitions (backtracks). - - Now `\d+` doesn't take all digits, but all except the last one: - ``` - \d+....... - (12345678)9z - ``` -5. Now the engine tries to continue the search from the new position (`9`). - - The star `pattern:(\d+)*` can be applied -- it gives the number `match:9`: - - ``` - - \d+.......\d+ - (12345678)(9)z - ``` - - The engine tries to match `$` again, but fails, because meets `subject:z`: - - ``` - X - \d+.......\d+ - (12345678)(9)z - ``` - - -5. There's no match, so the engine will continue backtracking, decreasing the number of repetitions for `pattern:\d+` down to 7 digits. So the rest of the string `subject:89` becomes the second `pattern:\d+`: - - ``` - X - \d+......\d+ - (1234567)(89)z - ``` - - ...Still no match for `pattern:$`. - - The search engine backtracks again. Backtracking generally works like this: the last greedy quantifier decreases the number of repetitions until it can. Then the previous greedy quantifier decreases, and so on. In our case the last greedy quantifier is the second `pattern:\d+`, from `subject:89` to `subject:8`, and then the star takes `subject:9`: - - ``` - X - \d+......\d+\d+ - (1234567)(8)(9)z - ``` -6. ...Fail again. The second and third `pattern:\d+` backtracked to the end, so the first quantifier shortens the match to `subject:123456`, and the star takes the rest: - - ``` - X - \d+.......\d+ - (123456)(789)z - ``` - - Again no match. The process repeats: the last greedy quantifier releases one character (`9`): - - ``` - X - \d+.....\d+ \d+ - (123456)(78)(9)z - ``` -7. ...And so on. - -The regular expression engine goes through all combinations of `123456789` and their subsequences. There are a lot of them, that's why it takes so long. - -What to do? - -Should we turn on the lazy mode? - -Unfortunately, it doesn't: if we replace `pattern:\d+` with `pattern:\d+?`, that still hangs: - -```js run -// sloooooowwwwww -alert( '12345678901234567890123456789123456789z'.match(/(\d+?)*$/) ); -``` - -Lazy quantifiers actually do the same, but in the reverse order. - -Just think about how the search engine would work in this case. - -Some regular expression engines have tricky built-in checks to detect infinite backtracking or other means to work around them, but there's no universal solution. - -## Back to tags - -In the example above, when we search `pattern:<(\s*\w+=\w+\s*)*>` in the string `subject:<a=b a=b a=b a=b` -- the similar thing happens. - -The string has no `>` at the end, so the match is impossible, but the regexp engine doesn't know about it. The search backtracks trying different combinations of `pattern:(\s*\w+=\w+\s*)`: - -``` -(a=b a=b a=b) (a=b) -(a=b a=b) (a=b a=b) -(a=b) (a=b a=b a=b) -... -``` - -## How to fix? - -The backtracking checks many variants that are an obvious fail for a human. - -For instance, in the pattern `pattern:(\d+)*$` a human can easily see that `pattern:(\d+)*` does not need to backtrack `pattern:+`. There's no difference between one or two `\d+`: - -``` -\d+........ -(123456789)z - -\d+...\d+.... -(1234)(56789)z -``` - -Let's get back to more real-life example: `pattern:<(\s*\w+=\w+\s*)*>`. We want it to find pairs `name=value` (as many as it can). - -What we would like to do is to forbid backtracking. - -There's totally no need to decrease the number of repetitions. - -In other words, if it found three `name=value` pairs and then can't find `>` after them, then there's no need to decrease the count of repetitions. There are definitely no `>` after those two (we backtracked one `name=value` pair, it's there): - -``` -(name=value) name=value -``` - -Modern regexp engines support so-called "possessive" quantifiers for that. They are like greedy, but don't backtrack at all. Pretty simple, they capture whatever they can, and the search continues. There's also another tool called "atomic groups" that forbid backtracking inside parentheses. - -Unfortunately, but both these features are not supported by JavaScript. - -### Lookahead to the rescue - -We can forbid backtracking using lookahead. - -The pattern to take as much repetitions as possible without backtracking is: `pattern:(?=(a+))\1`. - -In other words: -- The lookahead `pattern:?=` looks for the maximal count `pattern:a+` from the current position. -- And then they are "consumed into the result" by the backreference `pattern:\1` (`pattern:\1` corresponds to the content of the second parentheses, that is `pattern:a+`). - -There will be no backtracking, because lookahead does not backtrack. If it found like 5 times of `pattern:a+` and the further match failed, then it doesn't go back to 4. - -```smart -There's more about the relation between possessive quantifiers and lookahead in articles [Regex: Emulate Atomic Grouping (and Possessive Quantifiers) with LookAhead](http://instanceof.me/post/52245507631/regex-emulate-atomic-grouping-with-lookahead) and [Mimicking Atomic Groups](http://blog.stevenlevithan.com/archives/mimic-atomic-groups). -``` - -So this trick makes the problem disappear. - -Let's fix the regexp for a tag with attributes from the beginning of the chapter`pattern:<\w+(\s*\w+=(\w+|"[^"]*")\s*)*>`. We'll use lookahead to prevent backtracking of `name=value` pairs: - -```js run -// regexp to search name=value -let attrReg = /(\s*\w+=(\w+|"[^"]*")\s*)/ - -// use new RegExp to nicely insert its source into (?=(a+))\1 -let fixedReg = new RegExp(`<\\w+(?=(${attrReg.source}*))\\1>`, 'g'); - -let goodInput = '...<a test="<>" href="#">... <b>...'; - -let badInput = `<tag a=b a=b a=b a=b a=b a=b a=b a=b - a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b a=b`; - -alert( goodInput.match(fixedReg) ); // <a test="<>" href="#">, <b> -alert( badInput.match(fixedReg) ); // null (no results, fast!) -``` - -Great, it works! We found both a long tag `match:<a test="<>" href="#">` and a small one `match:<b>`, and (!) didn't hang the engine on the bad input. - -Please note the `attrReg.source` property. `RegExp` objects provide access to their source string in it. That's convenient when we want to insert one regexp into another. diff --git a/9-regular-expressions/16-regexp-sticky/article.md b/9-regular-expressions/16-regexp-sticky/article.md new file mode 100644 index 000000000..ece2c960f --- /dev/null +++ b/9-regular-expressions/16-regexp-sticky/article.md @@ -0,0 +1,138 @@ + +# Sticky flag "y", searching at position + +The flag `pattern:y` allows to perform the search at the given position in the source string. + +To grasp the use case of `pattern:y` flag, and better understand the ways of regexps, let's explore a practical example. + +One of common tasks for regexps is "lexical analysis": we get a text, e.g. in a programming language, and need to find its structural elements. For instance, HTML has tags and attributes, JavaScript code has functions, variables, and so on. + +Writing lexical analyzers is a special area, with its own tools and algorithms, so we don't go deep in there, but there's a common task: to read something at the given position. + +E.g. we have a code string `subject:let varName = "value"`, and we need to read the variable name from it, that starts at position `4`. + +We'll look for variable name using regexp `pattern:\w+`. Actually, JavaScript variable names need a bit more complex regexp for accurate matching, but here it doesn't matter. + +- A call to `str.match(/\w+/)` will find only the first word in the line (`let`). That's not it. +- We can add the flag `pattern:g`. But then the call `str.match(/\w+/g)` will look for all words in the text, while we need one word at position `4`. Again, not what we need. + +**So, how to search for a regexp exactly at the given position?** + +Let's try using method `regexp.exec(str)`. + +For a `regexp` without flags `pattern:g` and `pattern:y`, this method looks only for the first match, it works exactly like `str.match(regexp)`. + +...But if there's flag `pattern:g`, then it performs the search in `str`, starting from position stored in the `regexp.lastIndex` property. And, if it finds a match, then sets `regexp.lastIndex` to the index immediately after the match. + +In other words, `regexp.lastIndex` serves as a starting point for the search, that each `regexp.exec(str)` call resets to the new value ("after the last match"). That's only if there's `pattern:g` flag, of course. + +So, successive calls to `regexp.exec(str)` return matches one after another. + +Here's an example of such calls: + +```js run +let str = 'let varName'; // Let's find all words in this string +let regexp = /\w+/g; + +alert(regexp.lastIndex); // 0 (initially lastIndex=0) + +let word1 = regexp.exec(str); +alert(word1[0]); // let (1st word) +alert(regexp.lastIndex); // 3 (position after the match) + +let word2 = regexp.exec(str); +alert(word2[0]); // varName (2nd word) +alert(regexp.lastIndex); // 11 (position after the match) + +let word3 = regexp.exec(str); +alert(word3); // null (no more matches) +alert(regexp.lastIndex); // 0 (resets at search end) +``` + +We can get all matches in the loop: + +```js run +let str = 'let varName'; +let regexp = /\w+/g; + +let result; + +while (result = regexp.exec(str)) { + alert( `Found ${result[0]} at position ${result.index}` ); + // Found let at position 0, then + // Found varName at position 4 +} +``` + +Such use of `regexp.exec` is an alternative to method `str.matchAll`, with a bit more control over the process. + +Let's go back to our task. + +We can manually set `lastIndex` to `4`, to start the search from the given position! + +Like this: + +```js run +let str = 'let varName = "value"'; + +let regexp = /\w+/g; // without flag "g", property lastIndex is ignored + +*!* +regexp.lastIndex = 4; +*/!* + +let word = regexp.exec(str); +alert(word); // varName +``` + +Hooray! Problem solved! + +We performed a search of `pattern:\w+`, starting from position `regexp.lastIndex = 4`. + +The result is correct. + +...But wait, not so fast. + +Please note: the `regexp.exec` call starts searching at position `lastIndex` and then goes further. If there's no word at position `lastIndex`, but it's somewhere after it, then it will be found: + +```js run +let str = 'let varName = "value"'; + +let regexp = /\w+/g; + +*!* +// start the search from position 3 +regexp.lastIndex = 3; +*/!* + +let word = regexp.exec(str); +// found the match at position 4 +alert(word[0]); // varName +alert(word.index); // 4 +``` + +For some tasks, including the lexical analysis, that's just wrong. We need to find a match exactly at the given position at the text, not somewhere after it. And that's what the flag `y` is for. + +**The flag `pattern:y` makes `regexp.exec` to search exactly at position `lastIndex`, not "starting from" it.** + +Here's the same search with flag `pattern:y`: + +```js run +let str = 'let varName = "value"'; + +let regexp = /\w+/y; + +regexp.lastIndex = 3; +alert( regexp.exec(str) ); // null (there's a space at position 3, not a word) + +regexp.lastIndex = 4; +alert( regexp.exec(str) ); // varName (word at position 4) +``` + +As we can see, regexp `pattern:/\w+/y` doesn't match at position `3` (unlike the flag `pattern:g`), but matches at position `4`. + +Not only that's what we need, there's an important performance gain when using flag `pattern:y`. + +Imagine, we have a long text, and there are no matches in it, at all. Then a search with flag `pattern:g` will go till the end of the text and find nothing, and this will take significantly more time than the search with flag `pattern:y`, that checks only the exact position. + +In tasks like lexical analysis, there are usually many searches at an exact position, to check what we have there. Using flag `pattern:y` is the key for correct implementations and a good performance. diff --git a/9-regular-expressions/17-regexp-methods/article.md b/9-regular-expressions/17-regexp-methods/article.md new file mode 100644 index 000000000..897d0ffb6 --- /dev/null +++ b/9-regular-expressions/17-regexp-methods/article.md @@ -0,0 +1,361 @@ +# Methods of RegExp and String + +In this article we'll cover various methods that work with regexps in-depth. + +## str.match(regexp) + +The method `str.match(regexp)` finds matches for `regexp` in the string `str`. + +It has 3 modes: + +1. If the `regexp` doesn't have flag `pattern:g`, then it returns the first match as an array with capturing groups and properties `index` (position of the match), `input` (input string, equals `str`): + + ```js run + let str = "I love JavaScript"; + + let result = str.match(/Java(Script)/); + + alert( result[0] ); // JavaScript (full match) + alert( result[1] ); // Script (first capturing group) + alert( result.length ); // 2 + + // Additional information: + alert( result.index ); // 7 (match position) + alert( result.input ); // I love JavaScript (source string) + ``` + +2. If the `regexp` has flag `pattern:g`, then it returns an array of all matches as strings, without capturing groups and other details. + ```js run + let str = "I love JavaScript"; + + let result = str.match(/Java(Script)/g); + + alert( result[0] ); // JavaScript + alert( result.length ); // 1 + ``` + +3. If there are no matches, no matter if there's flag `pattern:g` or not, `null` is returned. + + That's an important nuance. If there are no matches, we don't get an empty array, but `null`. It's easy to make a mistake forgetting about it, e.g.: + + ```js run + let str = "I love JavaScript"; + + let result = str.match(/HTML/); + + alert(result); // null + alert(result.length); // Error: Cannot read property 'length' of null + ``` + + If we want the result to be an array, we can write like this: + + ```js + let result = str.match(regexp) || []; + ``` + +## str.matchAll(regexp) + +[recent browser="new"] + +The method `str.matchAll(regexp)` is a "newer, improved" variant of `str.match`. + +It's used mainly to search for all matches with all groups. + +There are 3 differences from `match`: + +1. It returns an iterable object with matches instead of an array. We can make a regular array from it using `Array.from`. +2. Every match is returned as an array with capturing groups (the same format as `str.match` without flag `pattern:g`). +3. If there are no results, it returns an empty iterable object instead of `null`. + +Usage example: + +```js run +let str = '<h1>Hello, world!</h1>'; +let regexp = /<(.*?)>/g; + +let matchAll = str.matchAll(regexp); + +alert(matchAll); // [object RegExp String Iterator], not array, but an iterable + +matchAll = Array.from(matchAll); // array now + +let firstMatch = matchAll[0]; +alert( firstMatch[0] ); // <h1> +alert( firstMatch[1] ); // h1 +alert( firstMatch.index ); // 0 +alert( firstMatch.input ); // <h1>Hello, world!</h1> +``` + +If we use `for..of` to loop over `matchAll` matches, then we don't need `Array.from` any more. + +## str.split(regexp|substr, limit) + +Splits the string using the regexp (or a substring) as a delimiter. + +We can use `split` with strings, like this: + +```js run +alert('12-34-56'.split('-')) // array of ['12', '34', '56'] +``` + +But we can split by a regular expression, the same way: + +```js run +alert('12, 34, 56'.split(/,\s*/)) // array of ['12', '34', '56'] +``` + +## str.search(regexp) + +The method `str.search(regexp)` returns the position of the first match or `-1` if none found: + +```js run +let str = "A drop of ink may make a million think"; + +alert( str.search( /ink/i ) ); // 10 (first match position) +``` + +**The important limitation: `search` only finds the first match.** + +If we need positions of further matches, we should use other means, such as finding them all with `str.matchAll(regexp)`. + +## str.replace(str|regexp, str|func) + +This is a generic method for searching and replacing, one of most useful ones. The swiss army knife for searching and replacing. + +We can use it without regexps, to search and replace a substring: + +```js run +// replace a dash by a colon +alert('12-34-56'.replace("-", ":")) // 12:34-56 +``` + +There's a pitfall though. + +**When the first argument of `replace` is a string, it only replaces the first match.** + +You can see that in the example above: only the first `"-"` is replaced by `":"`. + +To find all hyphens, we need to use not the string `"-"`, but a regexp `pattern:/-/g`, with the obligatory `pattern:g` flag: + +```js run +// replace all dashes by a colon +alert( '12-34-56'.replace( *!*/-/g*/!*, ":" ) ) // 12:34:56 +``` + +The second argument is a replacement string. We can use special characters in it: + +| Symbols | Action in the replacement string | +|--------|--------| +|`$&`|inserts the whole match| +|<code>$`</code>|inserts a part of the string before the match| +|`$'`|inserts a part of the string after the match| +|`$n`|if `n` is a 1-2 digit number, inserts the contents of n-th capturing group, for details see [](info:regexp-groups)| +|`$<name>`|inserts the contents of the parentheses with the given `name`, for details see [](info:regexp-groups)| +|`$$`|inserts character `$` | + +For instance: + +```js run +let str = "John Smith"; + +// swap first and last name +alert(str.replace(/(john) (smith)/i, '$2, $1')) // Smith, John +``` + +**For situations that require "smart" replacements, the second argument can be a function.** + +It will be called for each match, and the returned value will be inserted as a replacement. + +The function is called with arguments `func(match, p1, p2, ..., pn, offset, input, groups)`: + +1. `match` -- the match, +2. `p1, p2, ..., pn` -- contents of capturing groups (if there are any), +3. `offset` -- position of the match, +4. `input` -- the source string, +5. `groups` -- an object with named groups. + +If there are no parentheses in the regexp, then there are only 3 arguments: `func(str, offset, input)`. + +For example, let's uppercase all matches: + +```js run +let str = "html and css"; + +let result = str.replace(/html|css/gi, str => str.toUpperCase()); + +alert(result); // HTML and CSS +``` + +Replace each match by its position in the string: + +```js run +alert("Ho-Ho-ho".replace(/ho/gi, (match, offset) => offset)); // 0-3-6 +``` + +In the example below there are two parentheses, so the replacement function is called with 5 arguments: the first is the full match, then 2 parentheses, and after it (not used in the example) the match position and the source string: + +```js run +let str = "John Smith"; + +let result = str.replace(/(\w+) (\w+)/, (match, name, surname) => `${surname}, ${name}`); + +alert(result); // Smith, John +``` + +If there are many groups, it's convenient to use rest parameters to access them: + +```js run +let str = "John Smith"; + +let result = str.replace(/(\w+) (\w+)/, (...match) => `${match[2]}, ${match[1]}`); + +alert(result); // Smith, John +``` + +Or, if we're using named groups, then `groups` object with them is always the last, so we can obtain it like this: + +```js run +let str = "John Smith"; + +let result = str.replace(/(?<name>\w+) (?<surname>\w+)/, (...match) => { + let groups = match.pop(); + + return `${groups.surname}, ${groups.name}`; +}); + +alert(result); // Smith, John +``` + +Using a function gives us the ultimate replacement power, because it gets all the information about the match, has access to outer variables and can do everything. + +## str.replaceAll(str|regexp, str|func) + +This method is essentially the same as `str.replace`, with two major differences: + +1. If the first argument is a string, it replaces *all occurrences* of the string, while `replace` replaces only the *first occurrence*. +2. If the first argument is a regular expression without the `g` flag, there'll be an error. With `g` flag, it works the same as `replace`. + +The main use case for `replaceAll` is replacing all occurrences of a string. + +Like this: + +```js run +// replace all dashes by a colon +alert('12-34-56'.replaceAll("-", ":")) // 12:34:56 +``` + + +## regexp.exec(str) + +The `regexp.exec(str)` method returns a match for `regexp` in the string `str`. Unlike previous methods, it's called on a regexp, not on a string. + +It behaves differently depending on whether the regexp has flag `pattern:g`. + +If there's no `pattern:g`, then `regexp.exec(str)` returns the first match exactly as `str.match(regexp)`. This behavior doesn't bring anything new. + +But if there's flag `pattern:g`, then: +- A call to `regexp.exec(str)` returns the first match and saves the position immediately after it in the property `regexp.lastIndex`. +- The next such call starts the search from position `regexp.lastIndex`, returns the next match and saves the position after it in `regexp.lastIndex`. +- ...And so on. +- If there are no matches, `regexp.exec` returns `null` and resets `regexp.lastIndex` to `0`. + +So, repeated calls return all matches one after another, using property `regexp.lastIndex` to keep track of the current search position. + +In the past, before the method `str.matchAll` was added to JavaScript, calls of `regexp.exec` were used in the loop to get all matches with groups: + +```js run +let str = 'More about JavaScript at https://javascript.info'; +let regexp = /javascript/ig; + +let result; + +while (result = regexp.exec(str)) { + alert( `Found ${result[0]} at position ${result.index}` ); + // Found JavaScript at position 11, then + // Found javascript at position 33 +} +``` + +This works now as well, although for newer browsers `str.matchAll` is usually more convenient. + +**We can use `regexp.exec` to search from a given position by manually setting `lastIndex`.** + +For instance: + +```js run +let str = 'Hello, world!'; + +let regexp = /\w+/g; // without flag "g", lastIndex property is ignored +regexp.lastIndex = 5; // search from 5th position (from the comma) + +alert( regexp.exec(str) ); // world +``` + +If the regexp has flag `pattern:y`, then the search will be performed exactly at the position `regexp.lastIndex`, not any further. + +Let's replace flag `pattern:g` with `pattern:y` in the example above. There will be no matches, as there's no word at position `5`: + +```js run +let str = 'Hello, world!'; + +let regexp = /\w+/y; +regexp.lastIndex = 5; // search exactly at position 5 + +alert( regexp.exec(str) ); // null +``` + +That's convenient for situations when we need to "read" something from the string by a regexp at the exact position, not somewhere further. + +## regexp.test(str) + +The method `regexp.test(str)` looks for a match and returns `true/false` whether it exists. + +For instance: + +```js run +let str = "I love JavaScript"; + +// these two tests do the same +alert( *!*/love/i*/!*.test(str) ); // true +alert( str.search(*!*/love/i*/!*) != -1 ); // true +``` + +An example with the negative answer: + +```js run +let str = "Bla-bla-bla"; + +alert( *!*/love/i*/!*.test(str) ); // false +alert( str.search(*!*/love/i*/!*) != -1 ); // false +``` + +If the regexp has flag `pattern:g`, then `regexp.test` looks from `regexp.lastIndex` property and updates this property, just like `regexp.exec`. + +So we can use it to search from a given position: + +```js run +let regexp = /love/gi; + +let str = "I love JavaScript"; + +// start the search from position 10: +regexp.lastIndex = 10; +alert( regexp.test(str) ); // false (no match) +``` + +````warn header="Same global regexp tested repeatedly on different sources may fail" +If we apply the same global regexp to different inputs, it may lead to wrong result, because `regexp.test` call advances `regexp.lastIndex` property, so the search in another string may start from non-zero position. + +For instance, here we call `regexp.test` twice on the same text, and the second time fails: + +```js run +let regexp = /javascript/g; // (regexp just created: regexp.lastIndex=0) + +alert( regexp.test("javascript") ); // true (regexp.lastIndex=10 now) +alert( regexp.test("javascript") ); // false +``` + +That's exactly because `regexp.lastIndex` is non-zero in the second test. + +To work around that, we can set `regexp.lastIndex = 0` before each search. Or instead of calling methods on regexp, use string methods `str.match/search/...`, they don't use `lastIndex`. +```` diff --git a/9-regular-expressions/20-regexp-unicode/article.md b/9-regular-expressions/20-regexp-unicode/article.md deleted file mode 100644 index 68eebca8b..000000000 --- a/9-regular-expressions/20-regexp-unicode/article.md +++ /dev/null @@ -1,89 +0,0 @@ - -# Unicode: flag "u" - -The unicode flag `/.../u` enables the correct support of surrogate pairs. - -Surrogate pairs are explained in the chapter <info:string>. - -Let's briefly remind them here. In short, normally characters are encoded with 2 bytes. That gives us 65536 characters maximum. But there are more characters in the world. - -So certain rare characters are encoded with 4 bytes, like `𝒳` (mathematical X) or `😄` (a smile). - -Here are the unicode values to compare: - -| Character | Unicode | Bytes | -|------------|---------|--------| -| `a` | 0x0061 | 2 | -| `≈` | 0x2248 | 2 | -|`𝒳`| 0x1d4b3 | 4 | -|`𝒴`| 0x1d4b4 | 4 | -|`😄`| 0x1f604 | 4 | - -So characters like `a` and `≈` occupy 2 bytes, and those rare ones take 4. - -The unicode is made in such a way that the 4-byte characters only have a meaning as a whole. - -In the past JavaScript did not know about that, and many string methods still have problems. For instance, `length` thinks that here are two characters: - -```js run -alert('😄'.length); // 2 -alert('𝒳'.length); // 2 -``` - -...But we can see that there's only one, right? The point is that `length` treats 4 bytes as two 2-byte characters. That's incorrect, because they must be considered only together (so-called "surrogate pair"). - -Normally, regular expressions also treat "long characters" as two 2-byte ones. - -That leads to odd results, for instance let's try to find `pattern:[𝒳𝒴]` in the string `subject:𝒳`: - -```js run -alert( '𝒳'.match(/[𝒳𝒴]/) ); // odd result (wrong match actually, "half-character") -``` - -The result is wrong, because by default the regexp engine does not understand surrogate pairs. - -So, it thinks that `[𝒳𝒴]` are not two, but four characters: -1. the left half of `𝒳` `(1)`, -2. the right half of `𝒳` `(2)`, -3. the left half of `𝒴` `(3)`, -4. the right half of `𝒴` `(4)`. - -We can list them like this: - -```js run -for(let i=0; i<'𝒳𝒴'.length; i++) { - alert('𝒳𝒴'.charCodeAt(i)); // 55349, 56499, 55349, 56500 -}; -``` - -So it finds only the "left half" of `𝒳`. - -In other words, the search works like `'12'.match(/[1234]/)`: only `1` is returned. - -## The "u" flag - -The `/.../u` flag fixes that. - -It enables surrogate pairs in the regexp engine, so the result is correct: - -```js run -alert( '𝒳'.match(/[𝒳𝒴]/u) ); // 𝒳 -``` - -Let's see one more example. - -If we forget the `u` flag and occasionally use surrogate pairs, then we can get an error: - -```js run -'𝒳'.match(/[𝒳-𝒴]/); // SyntaxError: invalid range in character class -``` - -Normally, regexps understand `[a-z]` as a "range of characters with codes between codes of `a` and `z`. - -But without `u` flag, surrogate pairs are assumed to be a "pair of independant characters", so `[𝒳-𝒴]` is like `[<55349><56499>-<55349><56500>]` (replaced each surrogate pair with code points). Now we can clearly see that the range `56499-55349` is unacceptable, as the left range border must be less than the right one. - -Using the `u` flag makes it work right: - -```js run -alert( '𝒴'.match(/[𝒳-𝒵]/u) ); // 𝒴 -``` diff --git a/9-regular-expressions/21-regexp-unicode-properties/article.md b/9-regular-expressions/21-regexp-unicode-properties/article.md deleted file mode 100644 index c11914825..000000000 --- a/9-regular-expressions/21-regexp-unicode-properties/article.md +++ /dev/null @@ -1,86 +0,0 @@ - -# Unicode character properties \p - -[Unicode](https://en.wikipedia.org/wiki/Unicode), the encoding format used by JavaScript strings, has a lot of properties for different characters (or, technically, code points). They describe which "categories" character belongs to, and a variety of technical details. - -In regular expressions these can be set by `\p{…}`. And there must be flag `'u'`. - -For instance, `\p{Letter}` denotes a letter in any of language. We can also use `\p{L}`, as `L` is an alias of `Letter`, there are shorter aliases for almost every property. - -Here's the main tree of properties: - -- Letter `L`: - - lowercase `Ll`, modifier `Lm`, titlecase `Lt`, uppercase `Lu`, other `Lo` -- Number `N`: - - decimal digit `Nd`, letter number `Nl`, other `No`: -- Punctuation `P`: - - connector `Pc`, dash `Pd`, initial quote `Pi`, final quote `Pf`, open `Ps`, close `Pe`, other `Po` -- Mark `M` (accents etc): - - spacing combining `Mc`, enclosing `Me`, non-spacing `Mn` -- Symbol `S`: - - currency `Sc`, modifier `Sk`, math `Sm`, other `So` -- Separator `Z`: - - line `Zl`, paragraph `Zp`, space `Zs` -- Other `C`: - - control `Cc`, format `Cf`, not assigned `Cn`, private use `Co`, surrogate `Cs`. - -```smart header="More information" -Interested to see which characters belong to a property? There's a tool at <http://cldr.unicode.org/unicode-utilities/list-unicodeset> for that. - -You could also explore properties at [Character Property Index](http://unicode.org/cldr/utility/properties.jsp). - -For the full Unicode Character Database in text format (along with all properties), see <https://www.unicode.org/Public/UCD/latest/ucd/>. -``` - -There are also other derived categories, like: -- `Alphabetic` (`Alpha`), includes Letters `L`, plus letter numbers `Nl` (e.g. roman numbers Ⅻ), plus some other symbols `Other_Alphabetic` (`OAltpa`). -- `Hex_Digit` includes hexadimal digits: `0-9`, `a-f`. -- ...Unicode is a big beast, it includes a lot of properties. - -For instance, let's look for a 6-digit hex number: - -```js run -let reg = /\p{Hex_Digit}{6}/u; // flag 'u' is requireds - -alert("color: #123ABC".match(reg)); // 123ABC -``` - -There are also properties with a value. For instance, Unicode "Script" (a writing system) can be Cyrillic, Greek, Arabic, Han (Chinese) etc, the [list is long]("https://en.wikipedia.org/wiki/Script_(Unicode)"). - -To search for certain scripts, we should supply `Script=<value>`, e.g. to search for cyrillic letters: `\p{sc=Cyrillic}`, for Chinese glyphs: `\p{sc=Han}`, etc: - -```js run -let regexp = /\p{sc=Han}+/gu; // get chinese words - -let str = `Hello Привет 你好 123_456`; - -alert( str.match(regexp) ); // 你好 -``` - -## Building multi-language \w - -Let's make a "universal" regexp for `pattern:\w`, for any language. That task has a standard solution in many programming languages with unicode-aware regexps, e.g. Perl. - -```js -/[\p{Alphabetic}\p{Mark}\p{Decimal_Number}\p{Connector_Punctuation}\p{Join_Control}]/u -``` - -Let's decipher. Remember, `pattern:\w` is actually the same as `pattern:[a-zA-Z0-9_]`. - -So the character set includes: - -- `Alphabetic` for letters, -- `Mark` for accents, as in Unicode accents may be represented by separate code points, -- `Decimal_Number` for numbers, -- `Connector_Punctuation` for the `'_'` character and alike, -- `Join_Control` -– two special code points with hex codes `200c` and `200d`, used in ligatures e.g. in arabic. - -Or, if we replace long names with aliases (a list of aliases [here](https://www.unicode.org/Public/UCD/latest/ucd/PropertyValueAliases.txt)): - -```js run -let regexp = /([\p{Alpha}\p{M}\p{Nd}\p{Pc}\p{Join_C}]+)/gu; - -let str = `Hello Привет 你好 123_456`; - -alert( str.match(regexp) ); // Hello,Привет,你好,123_456 -``` diff --git a/9-regular-expressions/22-regexp-sticky/article.md b/9-regular-expressions/22-regexp-sticky/article.md deleted file mode 100644 index 5a790bb31..000000000 --- a/9-regular-expressions/22-regexp-sticky/article.md +++ /dev/null @@ -1,71 +0,0 @@ - -# Sticky flag "y", searching at position - -To grasp the use case of `y` flag, and see how great it is, let's explore a practical use case. - -One of common tasks for regexps is "parsing": when we get a text and analyze it for logical components, build a structure. - -For instance, there are HTML parsers for browser pages, that turn text into a structured document. There are parsers for programming languages, like JavaScript, etc. - -Writing parsers is a special area, with its own tools and algorithms, so we don't go deep in there, but there's a very common question: "What is the text at the given position?". - -For instance, for a programming language variants can be like: -- Is it a "name" `pattern:\w+`? -- Or is it a number `pattern:\d+`? -- Or an operator `pattern:[+-/*]`? -- (a syntax error if it's not anything in the expected list) - -In JavaScript, to perform a search starting from a given position, we can use `regexp.exec` with `regexp.lastIndex` property, but that's not what we need! - -We'd like to check the match exactly at given position, not "starting" from it. - -Here's a (failing) attempt to use `lastIndex`: - -```js run -let str = "(text before) function ..."; - -// attempting to find function at position 5: -let regexp = /function/g; // must use "g" flag, otherwise lastIndex is ignored -regexp.lastIndex = 5 - -alert (regexp.exec(str)); // function -``` - -The match is found, because `regexp.exec` starts to search from the given position and goes on by the text, successfully matching "function" later. - -We could work around that by checking if "`regexp.exec(str).index` property is `5`, and if not, ignore the much. But the main problem here is performance. - -The regexp engine does a lot of unnecessary work by scanning at further positions. The delays are clearly noticeable if the text is long, because there are many such searches in a parser. - -## The "y" flag - -So we've came to the problem: how to search for a match, starting exactly at the given position. - -That's what `y` flag does. It makes the regexp search only at the `lastIndex` position. - -Here's an example - -```js run -let str = "(text before) function ..."; - -*!* -let regexp = /function/y; -regexp.lastIndex = 5; -*/!* - -alert (regexp.exec(str)); // null (no match, unlike "g" flag!) - -*!* -regexp.lastIndex = 14; -*/!* - -alert (regexp.exec(str)); // function (match!) -``` - -As we can see, now the regexp is only matched at the given position. - -So what `y` does is truly unique, and very important for writing parsers. - -The `y` flag allows to apply a regular expression (or many of them one-by-one) exactly at the given position and when we understand what's there, we can move on -- step by step examining the text. - -Without the flag the regexp engine always searches till the end of the text, that takes time, especially if the text is large. So our parser would be very slow. The `y` flag is exactly the right thing here. diff --git a/BACKERS.md b/BACKERS.md new file mode 100644 index 000000000..36b1532bc --- /dev/null +++ b/BACKERS.md @@ -0,0 +1,6 @@ + +# Sponsors and Supporters + +## Supporters + +- Ilya Zelenko diff --git a/LICENSE.md b/LICENSE.md index 60f06924e..cbada5307 100644 --- a/LICENSE.md +++ b/LICENSE.md @@ -1,14 +1,143 @@ -# Attribution-NonCommercial-ShareAlike 4.0 +The tutorial is free to read. -The full license text is at <https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode>. +If you'd like to do something else with it, please get a permission from Ilya Kantor, iliakan@javascript.info. -You are free to: +As of now, we license the tutorial to almost everyone for free under the terms of an open license. Just please be so kind to contact me. + +## License (Short) + +The license is based on [CC-BY-NC](https://creativecommons.org/licenses/by-nc/4.0/legalcode). + +It gives the right to: - **Share** – copy and redistribute the tutorial in any medium or material. - **Adapt** – remix, transform, and build upon the material. Under the following terms: -- **Attribution** — You must give appropriate credit, provide a link to the license, and indicate if changes were made. You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use. +- **Attribution** — You must give appropriate credit. You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use. - **NonCommercial** — You may not use the material for commercial purposes. -- **ShareAlike** — If you remix, transform, or build upon the material, you must distribute your contributions under the same license as the original. + +## License (Legal) + +By exercising the Licensed Rights (defined below), You accept and agree to be bound by the terms and conditions of this license ("License"). To the extent this License may be interpreted as a contract, You are granted the Licensed Rights in consideration of Your acceptance of these terms and conditions, and the Licensor grants You such rights in consideration of benefits the Licensor receives from making the Licensed Material available under these terms and conditions. + +### Section 1 – Definitions. + +a. __Adapted Material__ means material subject to Copyright and Similar Rights that is derived from or based upon the Licensed Material and in which the Licensed Material is translated, altered, arranged, transformed, or otherwise modified in a manner requiring permission under the Copyright and Similar Rights held by the Licensor. For purposes of this License, where the Licensed Material is a musical work, performance, or sound recording, Adapted Material is always produced where the Licensed Material is synched in timed relation with a moving image. + +b. __Adapter's License__ means the license You apply to Your Copyright and Similar Rights in Your contributions to Adapted Material in accordance with the terms and conditions of this License. + +c. __Copyright and Similar Rights__ means copyright and/or similar rights closely related to copyright including, without limitation, performance, broadcast, sound recording, and Sui Generis Database Rights, without regard to how the rights are labeled or categorized. For purposes of this License, the rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights. + +d. __Effective Technological Measures__ means those measures that, in the absence of proper authority, may not be circumvented under laws fulfilling obligations under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996, and/or similar international agreements. + +e. __Exceptions and Limitations__ means fair use, fair dealing, and/or any other exception or limitation to Copyright and Similar Rights that applies to Your use of the Licensed Material. + +f. __Licensed Material__ means the artistic or literary work, database, or other material to which the Licensor applied this License. + +g. __Licensed Rights__ means the rights granted to You subject to the terms and conditions of this License, which are limited to all Copyright and Similar Rights that apply to Your use of the Licensed Material and that the Licensor has authority to license. + +h. __Licensor__ means the individual(s) or entity(ies) granting rights under this License. + +i. __NonCommercial__ means not primarily intended for or directed towards commercial advantage or monetary compensation. For purposes of this License, the exchange of the Licensed Material for other material subject to Copyright and Similar Rights by digital file-sharing or similar means is NonCommercial provided there is no payment of monetary compensation in connection with the exchange. + +j. __Share__ means to provide material to the public by any means or process that requires permission under the Licensed Rights, such as reproduction, public display, public performance, distribution, dissemination, communication, or importation, and to make material available to the public including in ways that members of the public may access the material from a place and at a time individually chosen by them. + +k. __Sui Generis Database Rights__ means rights other than copyright resulting from Directive 96/9/EC of the European Parliament and of the Council of 11 March 1996 on the legal protection of databases, as amended and/or succeeded, as well as other essentially equivalent rights anywhere in the world. + +l. __You__ means the individual or entity exercising the Licensed Rights under this License. Your has a corresponding meaning. + +### Section 2 – Scope. + +a. ___License grant.___ + + 1. Subject to the terms and conditions of this License, the Licensor hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive license to exercise the Licensed Rights in the Licensed Material to: + + A. reproduce and Share the Licensed Material, in whole or in part, for NonCommercial purposes only; and + + B. produce, reproduce, and Share Adapted Material for NonCommercial purposes only. + + 2. __Exceptions and Limitations.__ For the avoidance of doubt, where Exceptions and Limitations apply to Your use, this License does not apply, and You do not need to comply with its terms and conditions. + + 3. __Term.__ The term of this License is specified in Section 6(a). + + 4. __Media and formats; technical modifications allowed.__ The Licensor authorizes You to exercise the Licensed Rights in all media and formats whether now known or hereafter created, and to make technical modifications necessary to do so. The Licensor waives and/or agrees not to assert any right or authority to forbid You from making technical modifications necessary to exercise the Licensed Rights, including technical modifications necessary to circumvent Effective Technological Measures. For purposes of this License, simply making modifications authorized by this Section 2(a)(4) never produces Adapted Material. + + 5. __No endorsement.__ Nothing in this License constitutes or may be construed as permission to assert or imply that You are, or that Your use of the Licensed Material is, connected with, or sponsored, endorsed, or granted official status by, the Licensor or others designated to receive attribution as provided in Section 3. + +b. ___Other rights.___ + + 1. Moral rights, such as the right of integrity, are not licensed under this License, nor are publicity, privacy, and/or other similar personality rights; however, to the extent possible, the Licensor waives and/or agrees not to assert any such rights held by the Licensor to the limited extent necessary to allow You to exercise the Licensed Rights, but not otherwise. + + 2. Patent and trademark rights are not licensed under this License. + + 3. To the extent possible, the Licensor waives any right to collect royalties from You for the exercise of the Licensed Rights, whether directly or through a collecting society under any voluntary or waivable statutory or compulsory licensing scheme. In all other cases the Licensor expressly reserves any right to collect such royalties, including when the Licensed Material is used other than for NonCommercial purposes. + +### Section 3 – License Conditions. + +Your exercise of the Licensed Rights is expressly made subject to the following conditions. + +1. If You Share the Licensed Material (including in modified form), You must attribute the Licensor by adding: + + i. identification of the creator(s) of the Licensed Material and any others designated to receive attribution, in any reasonable manner requested by the Licensor (including by pseudonym if designated); + + ii. a URI or hyperlink to the Licensed Material to the extent reasonably practicable; + +2. You may satisfy the conditions in Section 3(1) in any reasonable manner based on the medium, means, and context in which You Share the Licensed Material. For example, it may be reasonable to satisfy the conditions by providing a URI or hyperlink to a resource that includes the required information. + +3. If requested by the Licensor, You must remove any of the information required by Section 3(1) to the extent reasonably practicable. + +### Section 4 – Sui Generis Database Rights. + +Where the Licensed Rights include Sui Generis Database Rights that apply to Your use of the Licensed Material: + +a. for the avoidance of doubt, Section 2(a)(1) grants You the right to extract, reuse, reproduce, and Share all or a substantial portion of the contents of the database for NonCommercial purposes only; + +b. if You include all or a substantial portion of the database contents in a database in which You have Sui Generis Database Rights, then the database in which You have Sui Generis Database Rights (but not its individual contents) is Adapted Material; and + +c. You must comply with the conditions in Section 3 if You Share all or a substantial portion of the contents of the database. + +For the avoidance of doubt, this Section 4 supplements and does not replace Your obligations under this License where the Licensed Rights include other Copyright and Similar Rights. + +### Section 5 – Disclaimer of Warranties and Limitation of Liability. + +a. __Unless otherwise separately undertaken by the Licensor, to the extent possible, the Licensor offers the Licensed Material as-is and as-available, and makes no representations or warranties of any kind concerning the Licensed Material, whether express, implied, statutory, or other. This includes, without limitation, warranties of title, merchantability, fitness for a particular purpose, non-infringement, absence of latent or other defects, accuracy, or the presence or absence of errors, whether or not known or discoverable. Where disclaimers of warranties are not allowed in full or in part, this disclaimer may not apply to You.__ + +b. __To the extent possible, in no event will the Licensor be liable to You on any legal theory (including, without limitation, negligence) or otherwise for any direct, special, indirect, incidental, consequential, punitive, exemplary, or other losses, costs, expenses, or damages arising out of this License or use of the Licensed Material, even if the Licensor has been advised of the possibility of such losses, costs, expenses, or damages. Where a limitation of liability is not allowed in full or in part, this limitation may not apply to You.__ + +c. The disclaimer of warranties and limitation of liability provided above shall be interpreted in a manner that, to the extent possible, most closely approximates an absolute disclaimer and waiver of all liability. + +### Section 6 – Term and Termination. + +a. This License applies for the term of the Copyright and Similar Rights licensed here. However, if You fail to comply with this License, then Your rights under this License terminate automatically. + +b. Where Your right to use the Licensed Material has terminated under Section 6(a), it reinstates: + + 1. automatically as of the date the violation is cured, provided it is cured within 30 days of Your discovery of the violation; or + + 2. upon express reinstatement by the Licensor. + + For the avoidance of doubt, this Section 6(b) does not affect any right the Licensor may have to seek remedies for Your violations of this License. + +c. Notwithstanding the above, the Licensor reserves the right to terminate this License with respect to You if the Licensor expressly notifies You of the termination. + +d. For the avoidance of doubt, the Licensor may also offer the Licensed Material under separate terms or conditions or stop distributing the Licensed Material at any time. + +e. Sections 1, 5, 6, 7, and 8 survive termination of this License. + +### Section 7 – Other Terms and Conditions. + +a. The Licensor shall not be bound by any additional or different terms or conditions communicated by You unless expressly agreed. + +b. Any arrangements, understandings, or agreements regarding the Licensed Material not stated herein are separate from and independent of the terms and conditions of this License. + +### Section 8 – Interpretation. + +a. For the avoidance of doubt, this License does not, and shall not be interpreted to, reduce, limit, restrict, or impose conditions on any use of the Licensed Material that could lawfully be made without permission under this License. + +b. To the extent possible, if any provision of this License is deemed unenforceable, it shall be automatically reformed to the minimum extent necessary to make it enforceable. If the provision cannot be reformed, it shall be severed from this License without affecting the enforceability of the remaining terms and conditions. + +c. No term or condition of this License will be waived and no failure to comply consented to unless expressly agreed to by the Licensor. + +d. Nothing in this License constitutes or may be interpreted as a limitation upon, or waiver of, any privileges and immunities that apply to the Licensor or You, including from the legal processes of any jurisdiction or authority. diff --git a/README.md b/README.md index 98983b623..cc2e55f67 100755 --- a/README.md +++ b/README.md @@ -5,6 +5,7 @@ Dette oppbevaringsstedet er vertsplass for oversettelsen av <https://javascript. **Slik kan du bidra:** +<<<<<<< HEAD - Sjekk ut [Den Norske Oversettelses fremdriften](https://github.com/javascript-tutorial/no.javascript.info/issues/1) issue. - Velg en umarkert artikkel du har lyst til å oversette. - Legg til en kommentar ved artikkelens navn i issue, f.eks `An Introduction to JavaScript`. @@ -14,28 +15,49 @@ Dette oppbevaringsstedet er vertsplass for oversettelsen av <https://javascript. - PR tittel burde være lik tittelen til artikkelen du oversetter, bot'en vil skrive nummeret til issue. Vær grei å la vedlikeholdere gå gjennom og merge, eller foreslå endringer i oversettelsen din. +======= +See <https://javascript.info/translate> for the details. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Hvis vedlikeholdere ikke svarer, eller du har lyst til å bli en vedlikeholder, kontakt oss i [hovedrepoet](https://github.com/javascript-tutorial/en.javascript.info/issues/new). **La andre få vite hva du oversetter, i meldingbordet, eller chat på ditt språk. Inviter de til å bli med!** +<<<<<<< HEAD 🎉 Tusen takk! Ditt navn og størrelsen på ditt bidrag vil dukke opp i "om prosjekt" delen av nettsiden når dette blir publisert. +======= +Something's wrong? A topic is missing? Explain it to people, add it as PR 👏 + +**You can edit the text in any editor.** The tutorial uses an enhanced "markdown" format, easy to grasp. And if you want to see how it looks on-site, there's a server to run the tutorial locally at <https://github.com/javascript-tutorial/server>. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b P.S. Hele listen av språk kan finnes her <https://javascript.info/translate>. ## Struktur +<<<<<<< HEAD Hvert kapittel, en artikkel eller oppgave holder til i sin egen mappe. Mappen sitt har navnet `N-url`, hvor `N` - står for nummer for hvordan (artiklene organiseres), og `url` er URL-stien på siden. +======= +Every chapter, article, or task has its folder. + +The folder is named like `N-url`, where `N` is a number for the sorting purposes and `URL` is the URL part with the title of the material. +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b Hver mappe har en av følgende filer: +<<<<<<< HEAD - `index.md` for en seksjon, - `article.md` for en artikkel, - `task.md` for en oppgave (+`solution.md` med løsningsteksten hvis det finnes en). +======= + - `index.md` stands for a chapter + - `article.md` stands for an article + - `task.md` stands for a task (solution must be provided in `solution.md` file as well) +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b En fil starter med `# Tittel overskrift`, og så er teksten skrevet i Markdown-lignende format, redigerbar i hvilken som helst text-editor. @@ -112,6 +134,7 @@ Noen filer, vanligvis oppgaver, har YAML metadata definert i toppen, markert med ```md importance: 5 +<<<<<<< HEAD --- ... ``` @@ -133,3 +156,8 @@ Vennligst ikke oversett eller fjern `[#...]` delen, dette er for URL ankerpunkte Du kan kjøre opplæringsdelens server lokalt for å se hvordan oversettelsene dine ser ut. Serveren og installasjonsinstrukser kan finnes her: <https://github.com/javascript-tutorial/server>. +======= +--- +♥ +Ilya Kantor @iliakan +>>>>>>> 540d753e90789205fc6e75c502f68382c87dea9b diff --git a/css.md b/css.md new file mode 100644 index 000000000..6a6316e69 --- /dev/null +++ b/css.md @@ -0,0 +1,4 @@ + +# CSS for JS developers + +- Outline diff --git a/figures.sketch b/figures.sketch index c740ac577..104333192 100644 Binary files a/figures.sketch and b/figures.sketch differ diff --git a/svgs.zip b/svgs.zip new file mode 100644 index 000000000..46eeb9862 Binary files /dev/null and b/svgs.zip differ