<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Rishav Sharan</title>
    <description>The latest articles on DEV Community by Rishav Sharan (@rishavs).</description>
    <link>https://dev.to/rishavs</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F107097%2F2a8f901d-c2c1-4665-bb75-5463eadb278c.jpeg</url>
      <title>DEV Community: Rishav Sharan</title>
      <link>https://dev.to/rishavs</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/rishavs"/>
    <language>en</language>
    <item>
      <title>Making a Single Page App in ye good olde JS (ES6)</title>
      <dc:creator>Rishav Sharan</dc:creator>
      <pubDate>Thu, 11 Oct 2018 12:58:04 +0000</pubDate>
      <link>https://dev.to/rishavs/making-a-single-page-app-in-ye-good-olde-js-es6-3eng</link>
      <guid>https://dev.to/rishavs/making-a-single-page-app-in-ye-good-olde-js-es6-3eng</guid>
      <description>&lt;p&gt;&lt;a href="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2F1uf9hnp02zsbjwxufync.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2F1uf9hnp02zsbjwxufync.png" alt="Imgur"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Demo: &lt;a href="https://src-brsetrrnrp.now.sh/" rel="noopener noreferrer"&gt;https://src-brsetrrnrp.now.sh/&lt;/a&gt;&lt;br&gt;
Repo: &lt;a href="https://github.com/rishavs/vanillajs-spa" rel="noopener noreferrer"&gt;https://github.com/rishavs/vanillajs-spa&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Making a SPA (single page app) is all the rage these days. A SPA is much snappier, gives better UX and can be much simpler to develop and deploy.&lt;br&gt;
But, to develop a SPA, one often requires lots of semantically huge frameworks like React, Vue etc and often, to use these frameworks properly, one needs more knowledge of these frameworks than one needs of the core language itself.&lt;/p&gt;

&lt;p&gt;With this article I want to show to you that you don't need to understand multiple ecosystems and frameworks to make simple frontend apps. Plain JS is all that you need. And I promise you that the overall app is going to be a whole lot simpler and easier to reason against.&lt;/p&gt;

&lt;p&gt;Few months ago, when I was making a weabpp in Crystal (love the language), I started with a simple server based webapp but soon realized that the amount of JS code I was writing was too much. My server codebase was actually much smaller. &lt;br&gt;
So, of course, I told myself that I might as well make the entire app in JS itself. &lt;/p&gt;

&lt;p&gt;(Note: this is a questionable logic, but I code for the fun of it and creating a simple SPA in just plain vanilla JS sounded like it would be the funnest idea ever! )&lt;br&gt;
("Narrator: No it wasn't").&lt;/p&gt;

&lt;p&gt;Surprisingly, there are not a lot of articles on making SPAs with vanillaJS and some of the articles that do guide the reader in this, tend to use a lot of libraries. Why am I stressing on  plain JS? This &lt;a href="http://vanilla-js.com/" rel="noopener noreferrer"&gt;page&lt;/a&gt; provides some compelling arguments.&lt;/p&gt;

&lt;p&gt;My plan was to use no library, unless it is clear to me that &lt;br&gt;
-- I am wasting too much time reinventing the wheel&lt;br&gt;
-- This is beyond me.&lt;/p&gt;

&lt;p&gt;So, handcoded Vanilla JS first. Libraries only when needed. &lt;br&gt;
Every bit of 3rd party code that you use comes with its own baggage and one should always check if the weight of baggage is less than the gains from using it.&lt;/p&gt;

&lt;p&gt;It also helps that I am making a simple blog/Hackernews type of app; with a list of posts in a page and the detailed post in another. Something that the new versions of JS (ES6) are well suited to handle.&lt;/p&gt;

&lt;p&gt;If you have to work on UX which needs real time updating of DOM, you really should stick to frameworks like React, Vue or (my personal favorite) Mithril.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;TLDR&lt;/strong&gt;; Simple and plain is good. Simple and plain is not &lt;em&gt;always&lt;/em&gt; good. &lt;br&gt;
Know your needs and the tools needed to best implement them.&lt;/p&gt;

&lt;p&gt;Before we begin with the implementation, let us look at what are we developing. &lt;/p&gt;

&lt;p&gt;The app will have the following pages:&lt;br&gt;
-- Home: containing a list of all posts. Shows data fetched from an api and rendered here.&lt;br&gt;
-- Post for specific id : containing details of that post. Shows the dynamic url parsing.&lt;br&gt;
-- About: Containing just some text. This is to showoff the router&lt;br&gt;
-- Secret : This page is not in my composed routes and will be used to show the 404 handling.&lt;br&gt;
-- Register: Contains a form and on clicking a button, shows the form data.&lt;/p&gt;

&lt;p&gt;Each page in my app itself will have the following structure:&lt;br&gt;
-- Navbar&lt;br&gt;
-- Content section&lt;br&gt;
-- Footer&lt;/p&gt;

&lt;p&gt;Each route in my app has the following structure:&lt;br&gt;
-- Resource&lt;br&gt;
-- Identifier&lt;br&gt;
-- verb&lt;/p&gt;

&lt;p&gt;For example, &lt;br&gt;
if the url is "localhost:8080/#/users/rishav/edit" then&lt;br&gt;
resource = "users", identifier = "rishav" and verb = "edit".&lt;/p&gt;

&lt;p&gt;Since I have  a very fixed structure for the urls that i will support, making a hash based router for it becomes real easy. In my route, the resource and verb strings are pre-defined but the identifier is dynamic.&lt;/p&gt;

&lt;p&gt;You can take a look at the &lt;a href="https://src-brsetrrnrp.now.sh/" rel="noopener noreferrer"&gt;demo app&lt;/a&gt; at:&lt;br&gt;
Do note, I have not made it pretty at all, as that's not the purpose of this article. Also, I am lazy :)&lt;/p&gt;



&lt;p&gt;Now that you have played a bit with it, lets get into the nitty-gritties.&lt;br&gt;
Currently, I am using &lt;a href="https://github.com/tapio/live-server" rel="noopener noreferrer"&gt;live-server&lt;/a&gt; for serving my spa in my dev environment. You can use any server that you want. In my examples here, i will refer to "localhost:8080" as my dev domain. &lt;br&gt;
I am also using &lt;a href="https://www.mockapi.io" rel="noopener noreferrer"&gt;https://www.mockapi.io&lt;/a&gt; to populate my app with sample data via a REST API.&lt;/p&gt;

&lt;p&gt;The article will focus on the following main aspects of the spa;&lt;br&gt;
-- The Router (using url hash) and&lt;br&gt;
-- Templating (using ES6 template literals) and&lt;br&gt;
-- The project architecture (using ES6 modules)&lt;/p&gt;
&lt;h3&gt;
  
  
  The Router
&lt;/h3&gt;

&lt;p&gt;The core idea behind the router is the use of the hash "#" in the urls. Whenever Browsers hit this character in a URL, they skip everything after it. So, from a browser's perspective "localhost:8080/#/nowhere" and "localhost:8080/#/somewhere" are the same and it will not send a server request to fetch the entire serverside route.&lt;br&gt;
You can read more on this at &lt;a href="https://en.wikipedia.org/wiki/Fragment_identifier" rel="noopener noreferrer"&gt;https://en.wikipedia.org/wiki/Fragment_identifier&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;The routing part of our code is;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;// List of supported routes. Any url other than these routes will throw a 404 error
const routes = {
    '/'             : Home
    , '/about'      : About
    , '/p/:id'      : PostShow
    , '/register'   : Register
};


// The router code. Takes a URL, checks against the list of supported routes and then renders the corresponding content page.
const router = async () =&amp;gt; {

    // Lazy load view element:
    const content = null || document.getElementById('page_container');

    // Get the parsed URl from the addressbar
    let request = Utils.parseRequestURL()

    // Parse the URL and if it has an id part, change it with the string ":id"
    let parsedURL = (request.resource ? '/' + request.resource : '/') + (request.id ? '/:id' : '') + (request.verb ? '/' + request.verb : '')

    // Get the page from our hash of supported routes.
    // If the parsed URL is not in our list of supported routes, select the 404 page instead
    let page = routes[parsedURL] ? routes[parsedURL] : Error404
    content.innerHTML = await page.render();
    await page.after_render();

}

// Listen on hash change:
window.addEventListener('hashchange', router);

// Listen on page load:
window.addEventListener('load', router);

&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;And in this code, I refer to a simple function in another file which is;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;    parseRequestURL : () =&amp;gt; {

        let url = location.hash.slice(1).toLowerCase() || '/';
        let r = url.split("/")
        let request = {
            resource    : null,
            id          : null,
            verb        : null
        }
        request.resource    = r[1]
        request.id          = r[2]
        request.verb        = r[3]

        return request
    }

&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Lets walk through what happens when the user puts the following url in the addressbar and clicks enter;&lt;br&gt;
-- localhost:8080/#/About&lt;/p&gt;

&lt;p&gt;First, this bit of code &lt;code&gt;window.addEventListener('load', router);&lt;/code&gt; gets fired as a browser load event is used.&lt;br&gt;
It then calls the router function.&lt;br&gt;
The router function first take the url from the addressbar and using the function &lt;code&gt;parseRequestURL&lt;/code&gt;, breaks it into our route schema of resource, identifier and verb. &lt;br&gt;
Then, this url is reformed by concatenating each url schema element.&lt;br&gt;
The final url string is then compared against an existing map of routes that we support.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;const routes = {
    '/'             : Home
    , '/about'      : About
    , '/p/:id'      : PostShow
    , '/register'   : Register
};
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Here each supported route is mapped against a content page.&lt;br&gt;
&lt;code&gt;let page = routes[parsedURL] ? routes[parsedURL] : Error404&lt;/code&gt;&lt;br&gt;
Then, we check if the parsed url string exists as a key in our routes map.&lt;br&gt;
if it does, the variable &lt;code&gt;page&lt;/code&gt; gets the corresponding value. Else it gets the value of the error404 page.&lt;/p&gt;

&lt;p&gt;In our example, the resource was "About", identifier was nil and verb was nil.&lt;br&gt;
So we check in the routes map for "about" and render that page using;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;    content.innerHTML = await page.render();

&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Lets now consider a dynamic route "localhost:8080/#/p/1234".&lt;/p&gt;

&lt;p&gt;Here the resource = "p", identifier ="1234" and the verb is nil.&lt;br&gt;
Since we have detected an identifier (which can be any string), we need to substitute it with a fixed string. So, i just replace any identifier with a fixed string ":id".&lt;br&gt;
This allows me to define a simple route as &lt;code&gt;"/p/1234" : PostShow&lt;/code&gt; where the PostShow page will show the data dynamically for the specific identifier.&lt;/p&gt;

&lt;p&gt;Of course we also need to ensure that all in-app links also have the "#" in them. For example, this is the html for the About page link in the navbar;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;&amp;lt;a class="navbar-item" href="/#/about"&amp;gt;
    About
&amp;lt;/a&amp;gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;When the user clicks on this link, the second global event that we defined &lt;code&gt;window.addEventListener('hashchange', router);&lt;/code&gt; gets fired. And then the router function is called again and yadda yadda and stuff.&lt;br&gt;
So, essentially, we re-render the page content of our app every time the user changes the url and loads the page (enter, refresh etc) or if they use an in-app hyperlink for navigation.&lt;/p&gt;

&lt;p&gt;Of course, our navigation state also gets saved in the browser history so you can use the back/forward browser buttons to navigate through the history state.&lt;/p&gt;

&lt;p&gt;What if a user tries a url like "localhost:1234/nowhere" (no # in the url at all after the origin)?&lt;br&gt;
This is not something that our purely clientside router can handle. In most hosting solutions for SPAs like Netifly, you can specify a 404 page (a pre-rendered version of the one we are using in our frontend app) which gets served in such scenarios.&lt;/p&gt;
&lt;h2&gt;
  
  
  Templating
&lt;/h2&gt;

&lt;p&gt;Lets take a look at a very simple page that we are generating, the About page;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;let About = {
    render : async () =&amp;gt; {
        let view =  /*html*/`
            &amp;lt;section class="section"&amp;gt;
                &amp;lt;h1&amp;gt; About &amp;lt;/h1&amp;gt;
            &amp;lt;/section&amp;gt;
        `
        return view
    }

}

export default About;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;I am using the ES6 template literals to define my html template. The &lt;code&gt;/*html*/&lt;/code&gt; bit is just a pragma that my VSCode extension uses to format the html content properly in the editor.&lt;br&gt;
The async bit doesn't makes much sense here but is needed for the pages that will be fetching and showing data from remote sources. I treat my About page a page template and every time i need to add a  new page to my app, i just do a bit of copy-paste-rename.&lt;br&gt;
If you noticed, the template literal string is inside a function. this means that the render function will be evaluated only when we need it.&lt;br&gt;
Earlier I had just done something like this;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;let About = /*html*/`
            &amp;lt;section class="section"&amp;gt;
                &amp;lt;h1&amp;gt; About &amp;lt;/h1&amp;gt;
            &amp;lt;/section&amp;gt;
            `

export default About;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This was a big problem as it would always evaluate this page no matter where I was in my app. If I had 20 pages in my app and many of these fetched some data, then no matter which area of the app i rendered, it would evaluate the entire app everytime (and fetch all the data for pages that I never needed).&lt;/p&gt;

&lt;p&gt;The functional approach also means that if we want to have small components in  a page, each component can have data passed to it for rendering using the function params. For example, if I want to have each link in my Home Page be a card instead, I can just call the render function for each link as &lt;code&gt;card.render(data)&lt;/code&gt;&lt;br&gt;
and read that data pro-\grammatically to render it accordingly.&lt;/p&gt;

&lt;p&gt;Now the next bit to tackle is adding interaction controls for our pages. For example, if our About page had a button; on clicking which a browser alert was raised, where should we put the code related to this button?&lt;/p&gt;

&lt;p&gt;Earlier, I had added a script tag in my string based html but no matter what I did, the code inside it wouldnt run.&lt;br&gt;
Finally a gentleman in another forum helped me out. It turns out that script tags added via the innerHTML method are not evaluated! Madness, I know! -__-&lt;br&gt;
Javascript is full of these small gotchas because of the security implications.&lt;/p&gt;

&lt;p&gt;So, that option was out.&lt;br&gt;
I also couldn't just add the actual code in a global scope and then refer to it by using onclick attribute on my button. &lt;/p&gt;

&lt;p&gt;The way to add event-handlers for your page controls is by adding another function to the page object like this;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;let About = {
    render : async () =&amp;gt; {
        let view =  /*html*/`
            &amp;lt;section class="section"&amp;gt;
                &amp;lt;h1&amp;gt; About &amp;lt;/h1&amp;gt;
                &amp;lt;button id="myBtn"&amp;gt; Button&amp;lt;/button&amp;gt;
            &amp;lt;/section&amp;gt;
        `
        return view
    },
    after_render: async () =&amp;gt; {
        document.getElementById("myBtn").addEventListener ("click",  () =&amp;gt; {
            console.log('Yo')
            alert('Yo')
        })
    }

}

export default About;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;and consuming it in the router as;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;    content.innerHTML = await page.render();
    await page.after_render();
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This allows me to define the code for the button press and this code gets initialized right after my dom is rendered.&lt;/p&gt;

&lt;h2&gt;
  
  
  App structure
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fknhhvz2cyg5lbrxachvh.png" class="article-body-image-wrapper"&gt;&lt;img src="https://media.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Farticles%2Fknhhvz2cyg5lbrxachvh.png" alt="Imgur"&gt;&lt;/a&gt;&lt;br&gt;
This is how I have structured the app. The views folder contains my dynamic content [pages]  and the smaller components [components]. If I add reusable components like cards, comments etc, I will be adding components for them here.&lt;/p&gt;

&lt;p&gt;If you have noticed, we are not using any bundler like parceljs, webpack etc because ES6 onward I really don't need to. I can use the simple html directive of script tag type=module to tell the browser that it we have a modularised app in our JS and it needs to consider the ES6 import/export commands and stitch the modules up accordingly.&lt;br&gt;
This is how we declare the root of our app in our core html file;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;    &amp;lt;script type="module" src="/app.js"&amp;gt;&amp;lt;/script&amp;gt;
&amp;lt;/head&amp;gt;

&amp;lt;body&amp;gt;
    &amp;lt;div id="header_container"&amp;gt;&amp;lt;/div&amp;gt;

    &amp;lt;div id="page_container" class="container pageEntry" &amp;gt;
        &amp;lt;article&amp;gt; Loading....&amp;lt;/article&amp;gt;
    &amp;lt;/div&amp;gt;

    &amp;lt;div id="footer_container"&amp;gt;&amp;lt;/div&amp;gt;
&amp;lt;/body&amp;gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;If you are curious about the "header_container"/"footer_container", these are the dom elements which contain my Navbar and the footer. The Navbar and Footer will be the same for each route and the content section will be dynamic and will change based on the currently selected route.&lt;/p&gt;

&lt;p&gt;I declare these components in the same way i declare and use my pages.&lt;/p&gt;

&lt;p&gt;For example, this is a Footer;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;let Bottombar = {
    render: async () =&amp;gt; {
        let view =  /*html*/`
        &amp;lt;footer class="footer"&amp;gt;
            &amp;lt;div class="content has-text-centered"&amp;gt;
                &amp;lt;p&amp;gt;
                    This is my foot. There are many like it, but this one is mine.
                &amp;lt;/p&amp;gt;
            &amp;lt;/div&amp;gt;
        &amp;lt;/footer&amp;gt;
        `
        return view
    },
    after_render: async () =&amp;gt; { }

}

export default Bottombar;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;and this is my modified Router where I consume this page;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;const router = async () =&amp;gt; {

    // Lazy load view element:
    const header = null || document.getElementById('header_container');
    const content = null || document.getElementById('page_container');
    const footer = null || document.getElementById('footer_container');

    // Render the Header and footer of the page
    header.innerHTML = await Navbar.render();
    await Navbar.after_render();
    footer.innerHTML = await Bottombar.render();
    await Bottombar.after_render();

&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;So no matter what route is entered via the url, i will always render the navbar and the footer.&lt;/p&gt;

&lt;p&gt;Of course, you may still want to use the bundlers to do stuff like minifying, gzipping your js files and they would be a great solution for it. I personally don't care about such optimizations right now and the sheer freedom from the whole webpack config &amp;gt; babel rabbit hole is immensely relieving.&lt;/p&gt;

&lt;p&gt;Whew. Look at the time!&lt;br&gt;
We are now done with our simple blog style spa!&lt;/p&gt;

&lt;p&gt;*&lt;em&gt;I would highly recommend you to go through the actual code at &lt;a href="https://github.com/rishavs/vanillajs-spa" rel="noopener noreferrer"&gt;https://github.com/rishavs/vanillajs-spa&lt;/a&gt; *&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;The code is simple, well commented and inline with what I wrote in this guide. Hopefully, it will help you get started on journey on a clean path through the cluttered JS framework jungle.&lt;/p&gt;

&lt;h3&gt;
  
  
  Credit:
&lt;/h3&gt;

&lt;p&gt;I myself have referenced a lot of guides and articles to be able to make this app.&lt;br&gt;
For the router, this one was a great help;&lt;br&gt;
&lt;a href="https://medium.com/@bryanmanuele/how-i-implemented-my-own-spa-routing-system-in-vanilla-js-49942e3c4573" rel="noopener noreferrer"&gt;https://medium.com/@bryanmanuele/how-i-implemented-my-own-spa-routing-system-in-vanilla-js-49942e3c4573&lt;/a&gt;&lt;/p&gt;

</description>
      <category>es6</category>
      <category>vanillajs</category>
      <category>spa</category>
      <category>singlepageapp</category>
    </item>
  </channel>
</rss>
