ساختن Google I/O 2016 Progressive Web App

خانه آیووا

خلاصه

بیاموزید که چگونه یک برنامه تک صفحه ای را با استفاده از اجزای وب، پلیمر و طراحی متریال ساختیم و آن را در Google.com به تولید رساندیم.

نتایج

  • تعامل بیشتر از برنامه بومی (4:06 دقیقه وب موبایل در مقابل اندروید 2:40 دقیقه).
  • 450 میلی‌ثانیه سریع‌تر رنگ‌آمیزی اولیه برای کاربران بازگشتی به لطف ذخیره‌سازی سرویس‌دهنده
  • 84 درصد از بازدیدکنندگان از Service Worker حمایت کردند
  • ذخیره‌های افزودن به صفحه اصلی در مقایسه با سال 2015 900٪ افزایش یافته است.
  • 3.8 درصد کاربران آفلاین شدند اما همچنان 11 هزار بازدید از صفحه ایجاد کردند!
  • 50 درصد از کاربرانی که وارد سیستم شده‌اند اعلان‌ها را فعال کرده‌اند.
  • 536 هزار اعلان برای کاربران ارسال شد (12 درصد آنها را بازگرداندند).
  • 99 درصد از مرورگرهای کاربران از polyfills اجزای وب پشتیبانی می کردند

نمای کلی

امسال، من این لذت را داشتم که روی برنامه وب پیشرو Google I/O 2016 کار کنم که با محبت "IOWA" نام داشت. ابتدا تلفن همراه است، کاملاً آفلاین کار می کند و به شدت از طراحی متریال الهام گرفته شده است.

IOWA یک برنامه کاربردی تک صفحه ای (SPA) است که با استفاده از کامپوننت های وب ، پلیمر، و Firebase ساخته شده است و دارای یک Backend گسترده است که در App Engine (Go) نوشته شده است. با استفاده از یک سرویس‌کار ، محتوا را از پیش ذخیره می‌کند، صفحات جدید را به‌صورت پویا بارگیری می‌کند، به‌راحتی بین نماها جابه‌جا می‌شود و پس از بارگیری مجدد از محتوا استفاده می‌کند.

در این مطالعه موردی، برخی از تصمیمات معماری جالب‌تری را که برای قسمت جلویی گرفته‌ایم مرور می‌کنم. اگر به کد منبع علاقه دارید، آن را در Github بررسی کنید .

مشاهده در Github

ساخت SPA با استفاده از اجزای وب

هر صفحه به عنوان یک جزء

یکی از جنبه های اصلی در مورد frontend ما این است که حول اجزای وب متمرکز شده است. در واقع، هر صفحه در SPA ما یک جزء وب است:

    <io-home-page date="2016-05-18T17:00:00Z" app="[[app]]"></io-home-page>
    <io-schedule-page date="2016-05-18T17:00:00Z" app="{ % templatetag openvariable % }app}}"></io-schedule-page>
    <io-attend-page></io-attend-page>
    <io-extended-page></io-extended-page>
    <io-faq-page></io-faq-page>

چرا این کار را کردیم؟ اولین دلیل این است که این کد قابل خواندن است. به عنوان اولین خواننده، کاملاً واضح است که هر صفحه در برنامه ما چیست. دلیل دوم این است که کامپوننت های وب ویژگی های خوبی برای ساخت SPA دارند. بسیاری از ناامیدی های رایج (مدیریت حالت، فعال سازی مشاهده، محدوده سبک) به لطف ویژگی های ذاتی عنصر <template> ، عناصر سفارشی و Shadow DOM از بین می روند. اینها ابزارهای توسعه دهنده ای هستند که در مرورگر ساخته می شوند. چرا از آنها سوء استفاده نمی کنید؟

با ایجاد یک عنصر سفارشی برای هر صفحه، چیزهای زیادی به صورت رایگان دریافت کردیم:

  • مدیریت چرخه عمر صفحه
  • محدوده CSS/HTML مخصوص صفحه.
  • تمام CSS/HTML/JS مختص یک صفحه بسته بندی شده و در صورت نیاز با هم بارگذاری می شوند.
  • نماها قابل استفاده مجدد هستند. از آنجایی که صفحات گره های DOM هستند، به سادگی افزودن یا حذف آن ها نمای را تغییر می دهد.
  • نگهبانان آینده می توانند برنامه ما را به سادگی با علامت گذاری درک کنند.
  • با ثبت و ارتقاء تعاریف عناصر توسط مرورگر، نشانه گذاری ارائه شده توسط سرور می تواند به تدریج افزایش یابد.
  • عناصر سفارشی یک مدل ارثی دارند. کد DRY کد خوبی است.
  • ... خیلی چیزهای بیشتر.

ما از این مزایا در IOWA نهایت استفاده را بردیم. بیایید به برخی از جزئیات شیرجه بزنیم.

فعال سازی پویا صفحات

عنصر <template> روش استاندارد مرورگر برای ایجاد نشانه گذاری قابل استفاده مجدد است. <template> دو ویژگی دارد که SPAها می توانند از آنها بهره ببرند. اول، هر چیزی در داخل <template> بی اثر است تا زمانی که نمونه ای از الگو ایجاد شود. دوم، مرورگر نشانه گذاری را تجزیه می کند اما محتوا از صفحه اصلی قابل دسترسی نیست. این یک تکه نشانه گذاری واقعی و قابل استفاده مجدد است. به عنوان مثال:

<template id="t">
    <div>This markup is inert and not part of the main page's DOM.</div>
    <img src="profile.png"> <!-- not loaded by the browser -->
    <video id="vid" src="vid.mp4"></video> <!-- doesn't load/start -->
    <script>alert("Not run until the template is stamped");</script>
</template>

پلیمر <template> را با چند عنصر سفارشی پسوند نوع گسترش می دهد ، یعنی <template is="dom-if"> و <template is="dom-repeat"> . هر دو عنصر سفارشی هستند که <template> با قابلیت های اضافی گسترش می دهند. و به لطف ماهیت اعلامی اجزای وب، هر دو دقیقاً همان کاری را انجام می دهند که انتظار دارید. مؤلفه اول نشانه گذاری را بر اساس شرطی مهر می کند. دومی نشانه گذاری را برای هر مورد در یک لیست (مدل داده) تکرار می کند.

IOWA چگونه از این المان‌های پسوندی استفاده می‌کند؟

اگر به خاطر داشته باشید، هر صفحه در IOWA یک جزء وب است. با این حال، احمقانه است که هر جزء را در بار اول اعلام کنیم . این به معنای ایجاد یک نمونه از هر صفحه در اولین بارگیری برنامه است. ما نمی‌خواستیم به عملکرد بارگذاری اولیه خود آسیبی وارد کنیم، به خصوص که برخی از کاربران فقط به 1 یا 2 صفحه پیمایش می‌کنند.

راه حل ما تقلب بود. در IOWA، عنصر هر صفحه را در یک <template is="dom-if"> قرار می دهیم تا محتویات آن در اولین بوت بارگذاری نشود. سپس زمانی که ویژگی name قالب با URL مطابقت داشته باشد، صفحات را فعال می کنیم. مؤلفه وب <lazy-pages> همه این منطق را برای ما مدیریت می کند. نشانه گذاری چیزی شبیه به این است:

<!-- Lazy pages manages the template stamping. It watches for route changes
        and sets `template.if = true` on the appropriate template. -->
<lazy-pages>
    <template is="dom-if" name="home">
    <io-home-page date="2016-05-18T17:00:00Z"></io-home-page>
    </template>

    <template is="dom-if" name="schedule">
    <io-schedule-page date="2016-05-18T17:00:00Z"></io-schedule-page>
    </template>

    <template is="dom-if" name="attend">
    <io-attend-page></io-attend-page>
    </template>
</lazy-pages>

چیزی که من در این مورد دوست دارم این است که هر صفحه وقتی صفحه بارگیری می شود، تجزیه می شود و آماده کار است، اما CSS/HTML/JS آن فقط بر اساس تقاضا اجرا می شود (زمانی که <template> مادر آن مهر شده باشد). نماهای پویا + تنبل با استفاده از اجزای وب FTW.

پیشرفت های آینده

هنگامی که صفحه برای اولین بار بارگیری می شود، ما در حال بارگیری همه واردات HTML برای هر صفحه به یکباره هستیم. یک پیشرفت آشکار این است که تعاریف عناصر را تنها در زمانی که به آنها نیاز است بارگذاری کنند. Polymer همچنین یک کمک کننده خوب برای بارگذاری ناهمگام واردات HTML دارد:

Polymer.Base.importHref('io-home-page.html', (e) => { ... });

IOWA این کار را انجام نمی دهد زیرا الف) ما تنبل شدیم و ب) مشخص نیست که چقدر افزایش عملکرد را شاهد بودیم. اولین رنگ ما قبلاً 1 ثانیه بود.

مدیریت چرخه عمر صفحه

Custom Elements API « بازخوانی چرخه حیات » را برای مدیریت وضعیت یک جزء تعریف می کند. هنگامی که این روش ها را پیاده سازی می کنید، قلاب های رایگان در طول عمر یک جزء دریافت می کنید:

createdCallback() {
    // automatically called when an instance of the element is created.
}

attachedCallback() {
    // automatically called when the element is attached to the DOM.
}

detachedCallback() {
    // automatically called when the element is removed from the DOM.
}

attributeChangedCallback() {
    // automatically called when an HTML attribute changes.
}

استفاده از این تماس ها در IOWA آسان بود. به یاد داشته باشید، هر صفحه یک گره DOM مستقل است. پیمایش به یک "نمای جدید" در SPA ما مربوط به اتصال یک گره به DOM و حذف یک گره دیگر است.

ما از attachedCallback برای انجام کار راه‌اندازی (وضعیت اولیه، پیوست شنوندگان رویداد) استفاده کردیم. هنگامی که کاربران به صفحه دیگری هدایت می‌شوند، detachedCallback پاکسازی می‌کند (حذف شنوندگان، بازنشانی حالت اشتراک‌گذاری شده). ما همچنین تماس‌های چرخه حیات بومی را با چندین مورد خود گسترش دادیم:

onPageTransitionDone() {
    // page transition animations are complete.
},

onSubpageTransitionDone() {
    // sub nav/tab page transitions are complete.
}

اینها اضافات مفیدی برای به تاخیر انداختن کار و به حداقل رساندن jank بین انتقال صفحه بودند. بیشتر در این مورد بعدا.

خشک کردن عملکرد رایج در سراسر صفحات

وراثت یکی از ویژگی های قدرتمند Custom Elements است. این یک مدل ارثی استاندارد برای وب فراهم می کند.

متأسفانه، پلیمر 1.0 هنوز در زمان نگارش وراثت عنصر را پیاده سازی نکرده است. در این بین، ویژگی Polymer's Behaviors به ​​همان اندازه مفید بود. رفتارها فقط ترکیب هستند.

به جای ایجاد سطح API یکسان در همه صفحات، خشک کردن پایگاه کد با ایجاد میکس های مشترک منطقی بود. برای مثال، PageBehavior ویژگی‌ها/روش‌های رایجی را که همه صفحات در برنامه ما به آن نیاز دارند، تعریف می‌کند:

PageBehavior.html

let PageBehavior = {

    // Common properties all pages need.
    properties: {
    name: { type: String }, // Slug name of the page.
    ...
    },

    attached() {
    // If the page defines a `onPageTransitionDone`, call it when the router
    // fires 'page-transition-done'.
    if (this.onPageTransitionDone) {
        this.listen(document.body, 'page-transition-done', 'onPageTransitionDone');
    }

    // Update page meta data when new page is navigated to.
    document.body.id = `page-${this.name}`;
    document.title = this.title || 'Google I/O 2016';

    // Scroll to top of new page.
    if (IOWA.Elements.Scroller) {
        IOWA.Elements.Scroller.scrollTop = 0;
    }

    this.setupSubnavEffects();
    },

    detached() {
    this.unlisten(document.body, 'page-transition-done', 'onPageTransitionDone');
    this.teardownSubnavEffects();
    }
};

IOWA.IOBehaviors = IOWA.IOBehaviors || {PageBehavior: PageBehavior};

همانطور که می بینید، PageBehavior وظایف معمولی را انجام می دهد که هنگام بازدید از یک صفحه جدید اجرا می شوند. مواردی مانند به‌روزرسانی document.title ، بازنشانی موقعیت اسکرول، و تنظیم شنوندگان رویداد برای جلوه‌های پیمایش و زیر پیمایش.

صفحات منفرد از PageBehavior با بارگذاری آن به عنوان یک وابستگی و استفاده از behaviors استفاده می کنند. آنها همچنین آزادند تا در صورت نیاز، ویژگی ها/روش های پایه آن را نادیده بگیرند. به عنوان مثال، در اینجا چیزی است که "subclass" صفحه اصلی ما لغو می شود:

io-home-page.html

<link rel="import" href="../bower_components/polymer/polymer.html">
<link rel="import" href="PageBehavior.html">
<!-- rest of the import dependencies used by the page. -->

<dom-module id="io-home-page">
    <template>
    <!-- PAGE'S MARKUP -->
    </template>
    <script>
    Polymer({
        is: 'io-home-page',

        behaviors: [IOBehaviors.PageBehavior], // All pages have common functionality.

        // Pages define their own title and slug for the router.
        title: 'Schedule - Google I/O 2016',
        name: 'home',

        // The home page has custom setup work when it's added navigated to.
        // Note: PageBehavior's attached also gets called.
        attached() {
        if (this.app.isPhoneSize) {
            this.listen(IOWA.Elements.ScrollContainer, 'scroll', '_onPageScroll');
        }
        },

        // The home page does its own cleanup when a new page is navigated to.
        // Note: PageBehavior's detached also gets called.
        detached() {
        this.unlisten(IOWA.Elements.ScrollContainer, 'scroll', '_onPageScroll');
        },

        // The home page can define onPageTransitionDone to do extra work
        // when page transitions are done, and thus preventing janky animations.
        onPageTransitionDone() {
        ...
        }
    });
    </script>
</dom-module>

به اشتراک گذاری سبک ها

برای اشتراک‌گذاری استایل‌ها در مؤلفه‌های مختلف در برنامه‌مان، از ماژول‌های سبک مشترک پلیمر استفاده کردیم. ماژول‌های Style به شما این امکان را می‌دهند که یک‌بار تکه‌ای از CSS را تعریف کنید و از آن در مکان‌های مختلف در سراسر برنامه استفاده مجدد کنید. برای ما «مکان های مختلف» به معنای اجزای مختلف بود.

در IOWA، ما shared-app-styles ایجاد کردیم تا رنگ‌ها، تایپوگرافی و کلاس‌های چیدمان را در صفحات و سایر اجزای سازنده به اشتراک بگذاریم.

shared-app-styles.html

<link rel="import" href="../bower_components/polymer/polymer.html">
<link rel="import" href="../bower_components/iron-flex-layout/iron-flex-layout.html">
<link rel="import" href="../bower_components/paper-styles/color.html">

<dom-module id="shared-app-styles">
    <template>
    <style>
        [layout] {
        @apply(--layout);
        }
        [layout][horizontal] {
        @apply(--layout-horizontal);
        }
        .scrollable {
        @apply(--layout-scroll);
        }
        .noscroll {
        overflow: hidden;
        }
        /* Style radio buttons and tabs the same throughout the app */
        paper-tabs {
        --paper-tabs-selection-bar-color: currentcolor;
        }
        paper-radio-button {
        --paper-radio-button-checked-color: var(--paper-cyan-600);
        --paper-radio-button-checked-ink-color: var(--paper-cyan-600);
        }
        ...
    </style>
    </template>
</dom-module>

io-home-page.html

<link rel="import" href="shared-app-styles.html">
<!-- Rest of import dependencies used by the page. -->

<dom-module id="io-home-page">
    <template>
    <style include="shared-app-styles">
        :host { display: block} /* Other element styles can go here. */
    </style>
    <!-- PAGE'S MARKUP -->
    </template>
    <script>Polymer({...});</script>
</dom-module>

در اینجا، <style include="shared-app-styles"></style> نحو Polymer برای گفتن "شامل سبک ها در ماژول به نام "shared-app-styles" است.

به اشتراک گذاری وضعیت برنامه

اکنون می دانید که هر صفحه در برنامه ما یک عنصر سفارشی است. میلیون بار گفتم خوب، اما اگر هر صفحه جزء وب مستقل باشد، ممکن است از خود بپرسید که چگونه وضعیت را در برنامه به اشتراک می گذاریم.

IOWA از تکنیکی مشابه تزریق وابستگی (Angular) یا redux (React) برای به اشتراک گذاری وضعیت استفاده می کند. ما یک ویژگی app جهانی ایجاد کردیم و ویژگی های فرعی مشترک را از آن آویزان کردیم. app با تزریق آن به هر مؤلفه ای که به داده های آن نیاز دارد به برنامه ما منتقل می شود. استفاده از ویژگی های اتصال داده پلیمر این کار را آسان می کند زیرا می توانیم سیم کشی را بدون نوشتن هیچ کدی انجام دهیم:

<lazy-pages>
    <template is="dom-if" name="home">
    <io-home-page date="2016-05-18T17:00:00Z" app="[[app]]"></io-home-page>
    </template>

    <template is="dom-if" name="schedule">
    <io-schedule-page date="2016-05-18T17:00:00Z" app="{ % templatetag openvariable % }app}}"></io-schedule-page>
    </template>
    ...
</lazy-pages>

<google-signin client-id="..." scopes="profile email"
                            user="{ % templatetag openvariable % }app.currentUser}}"></google-signin>

<iron-media-query query="(min-width:320px) and (max-width:768px)"
                                query-matches="{ % templatetag openvariable % }app.isPhoneSize}}"></iron-media-query>

عنصر <google-signin> هنگامی که کاربران به برنامه ما وارد می شوند، ویژگی user خود را به روز می کند. از آنجایی که این ویژگی به app.currentUser محدود شده است، هر صفحه ای که می خواهد به کاربر فعلی دسترسی داشته باشد، به سادگی باید به app متصل شود و ویژگی فرعی currentUser را بخواند. این تکنیک به خودی خود برای به اشتراک گذاری وضعیت در سراسر برنامه مفید است. با این حال، یک مزیت دیگر این بود که ما در نهایت یک عنصر ورود به سیستم ایجاد کردیم و از نتایج آن در سراسر سایت استفاده مجدد کردیم . برای پرسش های رسانه ای هم همینطور. کپی کردن ورود به سیستم یا ایجاد مجموعه ای از درخواست های رسانه ای خود برای هر صفحه بیهوده بود. در عوض، اجزایی که مسئول عملکرد/داده های گسترده برنامه هستند در سطح برنامه وجود دارند.

انتقال صفحه

همانطور که در برنامه وب Google I/O پیمایش می کنید، متوجه تغییر صفحه نرم آن خواهید شد ( آلا طراحی متریال ).

انتقال صفحه IOWA در عمل.
انتقال صفحه IOWA در عمل.

هنگامی که کاربران به یک صفحه جدید هدایت می شوند، یک سری چیزها اتفاق می افتد:

  1. پیمایش بالای نوار انتخاب را به پیوند جدید می کشاند.
  2. عنوان صفحه محو می شود.
  3. محتوای صفحه به پایین می لغزد و سپس محو می شود.
  4. با معکوس کردن آن انیمیشن ها، عنوان و محتوای صفحه جدید ظاهر می شود.
  5. (اختیاری) صفحه جدید کارهای اولیه اضافی را انجام می دهد.

یکی از چالش‌های ما این بود که بفهمیم چگونه می‌توان این انتقال نرم را بدون قربانی کردن عملکرد ایجاد کرد. کارهای پویای زیادی وجود دارد که اتفاق می افتد و جنک در مهمانی ما مورد استقبال قرار نگرفت. راه حل ما ترکیبی از Web Animations API و Promises بود. استفاده از این دو با هم به ما تطبیق پذیری، سیستم پویانمایی plug and play و کنترل گرانول برای به حداقل رساندن das jank داد.

چگونه کار می کند

هنگامی که کاربران روی یک صفحه جدید کلیک می کنند (یا به عقب/ جلو می زنند)، runPageTransition() روتر ما جادوی خود را با اجرای یک سری Promises انجام می دهد. استفاده از Promises به ما این امکان را داد که انیمیشن ها را با دقت هماهنگ کنیم و به منطقی کردن «ناهمگام بودن» انیمیشن های CSS و بارگذاری پویا محتوا کمک کرد.

class Router {

    init() {
    window.addEventListener('popstate', e => this.runPageTransition());
    }

    runPageTransition() {
    let endPage = this.state.end.page;

    this.fire('page-transition-start');              // 1. Let current page know it's starting.

    IOWA.PageAnimation.runExitAnimation()            // 2. Play exist animation sequence.
        .then(() => {
        IOWA.Elements.LazyPages.selected = endPage;  // 3. Activate new page in <lazy-pages>.
        this.state.current = this.parseUrl(this.state.end.href);
        })
        .then(() => IOWA.PageAnimation.runEnterAnimation())  // 4. Play entry animation sequence.
        .then(() => this.fire('page-transition-done')) // 5. Tell new page transitions are done.
        .catch(e => IOWA.Util.reportError(e));
    }

}

از بخش "DRY نگه داشتن چیزها: عملکرد مشترک در بین صفحات" را به خاطر بیاورید ، صفحات به رویدادهای DOM page-transition-start و page-transition-done گوش می دهند. اکنون شما می بینید که آن رویدادها از کجا شروع می شوند.

ما از Web Animations API به جای راهنماهای runEnterAnimation / runExitAnimation استفاده کردیم. در مورد runExitAnimation ، ما چند گره DOM (منطقه مستطیل و محتوای اصلی) را می گیریم، شروع/پایان هر انیمیشن را اعلام می کنیم و یک GroupEffect ایجاد می کنیم تا این دو به صورت موازی اجرا شوند:

function runExitAnimation(section) {
    let main = section.querySelector('.slide-up');
    let masthead = section.querySelector('.masthead');

    let start = {transform: 'translate(0,0)', opacity: 1};
    let end = {transform: 'translate(0,-100px)', opacity: 0};
    let opts = {duration: 400, easing: 'cubic-bezier(.4, 0, .2, 1)'};
    let opts_delay = {duration: 400, delay: 200};

    return new GroupEffect([
    new KeyframeEffect(masthead, [start, end], opts),
    new KeyframeEffect(main, [{opacity: 1}, {opacity: 0}], opts_delay)
    ]);
}

فقط آرایه را تغییر دهید تا انتقال های مشاهده بیشتر (یا کمتر) دقیق تر شود!

جلوه های اسکرول

هنگامی که صفحه را اسکرول می کنید، IOWA چند جلوه جالب دارد. اولین مورد ، دکمه اکشن شناور ما (FAB) است که کاربران را به بالای صفحه برمی گرداند:

    <a href="#" tabindex="-1" aria-hidden="true" aria-label="back to top" onclick="backToTop">
      <paper-fab icon="io:expand-less" noink tabindex="-1"></paper-fab>
    </a>

اسکرول صاف با استفاده از عناصر طرح‌بندی برنامه پلیمر اجرا می‌شود. آنها جلوه‌های اسکرول خارج از جعبه را ارائه می‌کنند، مانند نوارهای پیمایش چسبنده/بازگردان، سایه‌های رها، انتقال رنگ و پس‌زمینه، جلوه‌های اختلاف منظر، و پیمایش صاف.

    // Smooth scrolling the back to top FAB.
    function backToTop(e) {
      e.preventDefault();

      Polymer.AppLayout.scroll({top: 0, behavior: 'smooth',
                                target: document.documentElement});

      e.target.blur();  // Kick focus back to the page so user starts from the top of the doc.
    }

مکان دیگری که ما از عناصر <app-layout> استفاده کردیم، برای ناوبری چسبنده بود. همانطور که در ویدیو می بینید، وقتی کاربران صفحه را به پایین اسکرول می کنند، از بین می رود و هنگام اسکرول کردن به بالا برمی گردد.

نوارهای پیمایش چسبنده
اسکرول چسبنده با استفاده از .

ما از عنصر <app-header> تقریباً همانطور که هست استفاده کردیم. رها کردن و به دست آوردن جلوه های اسکرول فانتزی در برنامه آسان بود. مطمئناً، ما خودمان می‌توانستیم آن‌ها را پیاده‌سازی کنیم، اما داشتن جزئیات از قبل در یک جزء قابل استفاده مجدد، صرفه‌جویی زیادی در زمان داشت.

عنصر را اعلام کنید. آن را با ویژگی ها سفارشی کنید. شما تمام شده اید!

    <app-header reveals condenses effects="fade-background waterfall"></app-header>

نتیجه گیری

برای برنامه وب پیشرو I/O، به لطف اجزای وب و ابزارک های طراحی متریال از پیش ساخته پلیمر، توانستیم یک صفحه اصلی را در چند هفته بسازیم. ویژگی های API های بومی (Custom Elements، Shadow DOM، <template> ) به طور طبیعی به پویایی یک SPA کمک می کند. قابلیت استفاده مجدد باعث صرفه جویی در زمان می شود.

اگر علاقه مند به ایجاد یک برنامه وب پیشرو خود هستید، جعبه ابزار برنامه را بررسی کنید. Polymer's App Toolbox مجموعه ای از کامپوننت ها، ابزارها و قالب ها برای ساختن PWA با پلیمر است. این یک راه آسان برای بلند شدن و دویدن است.