It looks like this is a web page, not a feed. I looked for a feed associated with this page, but couldn't find one. Please enter the address of your feed to validate.

Source: https://studentprojectcode.com/feed.xml

  1. <!DOCTYPE html><html lang="en"> <head><meta charset="utf-8"><link rel="icon" type="image/svg+xml" href="/favicon.svg"><link rel="icon" type="image/png" href="/favicon.png"><link rel="icon" type="image/x-icon" href="/favicon.ico"><meta name="viewport" content="width=device-width"><link rel="canonical" href="https://studentprojectcode.com/blog/how-to-make-collapsible-or-expandable-views-in"><title>How to Make Collapsible Or Expandable Views In Swift?</title><meta name="title" content="How to Make Collapsible Or Expandable Views In Swift in 2025?"><meta name="description" content="Learn how to create collapsible or expandable views in Swift effortlessly with this comprehensive guide."><link rel="sitemap" href="/sitemap-index.xml"><link rel="alternate" type="application/rss+xml" title="How to Make Collapsible Or Expandable Views In Swift?" href="https://studentprojectcode.com/feed.xml"><meta property="og:title" content="How to Make Collapsible Or Expandable Views In Swift?"><meta property="og:description" content="Learn how to create collapsible or expandable views in Swift effortlessly with this comprehensive guide."><meta property="og:url" content="https://studentprojectcode.com/blog/how-to-make-collapsible-or-expandable-views-in"><meta property="og:image" content="https://cdn.blogweb.me/1/6555113c41b85bf4eb4c153f_d9e8cfc84a.png"><meta property="twitter:card" content="summary_large_image"><meta property="twitter:url" content="https://studentprojectcode.com/blog/how-to-make-collapsible-or-expandable-views-in"><meta property="twitter:title" content="How to Make Collapsible Or Expandable Views In Swift?"><meta property="twitter:description" content="Learn how to create collapsible or expandable views in Swift effortlessly with this comprehensive guide."><meta property="twitter:image" content="https://cdn.blogweb.me/1/6555113c41b85bf4eb4c153f_d9e8cfc84a.png"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLucXtAOvWDSHFF.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLufntAOvWDSHFF.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLucHtAOvWDSA.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VFteOYktMqlap.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VGdeOYktMqlap.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VF9eOYktMqg.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJbecnFHGPezSQ.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJnecnFHGPezSQ.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z11lFd2JQEl8qw.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z1JlFd2JQEl8qw.woff2" type="font/woff2"><link as="font" crossorigin rel="preload" href="https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z1xlFd2JQEk.woff2" type="font/woff2"><style>@font-face {unicode-range: U+0900-097F, U+1CD0-1CF9, U+200C-200D, U+20A8, U+20B9, U+20F0, U+25CC, U+A830-A839, U+A8E0-A8FF, U+11B00-11B09;font-weight: 400;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLucXtAOvWDSHFF.woff2)} @font-face {unicode-range: U+0100-02BA, U+02BD-02C5, U+02C7-02CC, U+02CE-02D7, U+02DD-02FF, U+0304, U+0308, U+0329, U+1D00-1DBF, U+1E00-1E9F, U+1EF2-1EFF, U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;font-weight: 400;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLufntAOvWDSHFF.woff2)} @font-face {unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;font-weight: 400;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiGyp8kv8JHgFVrJJLucHtAOvWDSA.woff2)} @font-face {unicode-range: U+0900-097F, U+1CD0-1CF9, U+200C-200D, U+20A8, U+20B9, U+20F0, U+25CC, U+A830-A839, U+A8E0-A8FF, U+11B00-11B09;font-weight: 700;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VFteOYktMqlap.woff2)} @font-face {unicode-range: U+0100-02BA, U+02BD-02C5, U+02C7-02CC, U+02CE-02D7, U+02DD-02FF, U+0304, U+0308, U+0329, U+1D00-1DBF, U+1E00-1E9F, U+1EF2-1EFF, U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;font-weight: 700;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VGdeOYktMqlap.woff2)} @font-face {unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;font-weight: 700;font-style: italic;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiDyp8kv8JHgFVrJJLmy15VF9eOYktMqg.woff2)} @font-face {unicode-range: U+0900-097F, U+1CD0-1CF9, U+200C-200D, U+20A8, U+20B9, U+20F0, U+25CC, U+A830-A839, U+A8E0-A8FF, U+11B00-11B09;font-weight: 400;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJbecnFHGPezSQ.woff2)} @font-face {unicode-range: U+0100-02BA, U+02BD-02C5, U+02C7-02CC, U+02CE-02D7, U+02DD-02FF, U+0304, U+0308, U+0329, U+1D00-1DBF, U+1E00-1E9F, U+1EF2-1EFF, U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;font-weight: 400;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJnecnFHGPezSQ.woff2)} @font-face {unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;font-weight: 400;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiEyp8kv8JHgFVrJJfecnFHGPc.woff2)} @font-face {unicode-range: U+0900-097F, U+1CD0-1CF9, U+200C-200D, U+20A8, U+20B9, U+20F0, U+25CC, U+A830-A839, U+A8E0-A8FF, U+11B00-11B09;font-weight: 600;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z11lFd2JQEl8qw.woff2)} @font-face {unicode-range: U+0100-02BA, U+02BD-02C5, U+02C7-02CC, U+02CE-02D7, U+02DD-02FF, U+0304, U+0308, U+0329, U+1D00-1DBF, U+1E00-1E9F, U+1EF2-1EFF, U+2020, U+20A0-20AB, U+20AD-20C0, U+2113, U+2C60-2C7F, U+A720-A7FF;font-weight: 600;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z1JlFd2JQEl8qw.woff2)} @font-face {unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;font-weight: 600;font-style: normal;font-family: 'Poppins';font-display: swap;src: url(https://fonts.gstatic.com/s/poppins/v24/pxiByp8kv8JHgFVrLEj6Z1xlFd2JQEk.woff2)} body { font-family: 'Poppins', '_font_fallback_503529998705', sans-serif; } @font-face { font-family: '_font_fallback_503529998705'; size-adjust: 100.00%; src: local('Arial'); ascent-override: 105.00%; descent-override: 35.00%; line-gap-override: 10.00%; }</style><script data-default-mode="light">
  2.  window.mode ??= (() => {
  3.    const defaultMode = document.currentScript.getAttribute('data-default-mode')
  4.    const storageKey = 'mode'
  5.    const store =
  6.      typeof localStorage !== 'undefined'
  7.        ? localStorage
  8.        : { getItem: () => null, setItem: () => {} }
  9.  
  10.    const mediaMatcher = window.matchMedia('(prefers-color-scheme: light)')
  11.  
  12.    mediaMatcher.addEventListener('change', () => {
  13.      applyMode(mode.getMode())
  14.    })
  15.  
  16.    function applyMode(mode) {
  17.      document.documentElement.dataset.mode = mode
  18.      document.documentElement.style.colorScheme = mode
  19.    }
  20.  
  21.    function setMode(mode = defaultMode) {
  22.      store.setItem(storageKey, mode)
  23.      applyMode(mode)
  24.    }
  25.  
  26.    function getMode() {
  27.      return store.getItem(storageKey) || defaultMode
  28.    }
  29.  
  30.    return { setMode, getMode }
  31.  })()
  32.  
  33.  mode.setMode(mode.getMode())
  34. </script> <script type="module">document.addEventListener("astro:after-swap",()=>window.mode.setMode(window.mode.getMode()));</script><meta name="astro-view-transitions-enabled" content="true"><meta name="astro-view-transitions-fallback" content="animate"><script type="module" src="/_astro/ClientRouter.astro_astro_type_script_index_0_lang.CtSceO8m.js"></script><link rel="stylesheet" href="/_astro/_page_.DL7IbyIm.css">
  35. <link rel="stylesheet" href="/_astro/index.DyFFQt9L.css"><script type="module" src="/_astro/page.JlAKmP-M.js"></script></head> <body>  <div class="mx-auto flex min-h-[100svh] w-full max-w-[90%] flex-col justify-between md:max-w-3xl"> <header> <a href="#main" class="absolute -top-full left-8 px-2 py-2 focus:top-8">Skip to main content</a> <div class="flex justify-between py-4 sm:py-8"> <a href="/" class="text-accent flex items-center gap-2 text-4xl font-semibold"> <span class="inline-block text-4xl font-extrabold text-transparent bg-clip-text bg-gradient-to-r from-orange-500 via-yellow-500 to-pink-500"> studentprojectcode.com </span> </a> <nav class="hidden flex-col justify-end sm:flex"> <ul class="flex gap-6"> <li class="flex flex-col justify-center"> <a href="/blog" class="flex items-center gap-1 whitespace-nowrap text-accent/100" target="_self">  <span class="underline"> Blog </span> </a> </li> <li class="text-2xl"> <a href="/search/" class="flex items-center" aria-label="Search"> <svg width="1em" height="1em" class="clickable" data-icon="tabler:search">   <symbol id="ai:tabler:search" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10a7 7 0 1 0 14 0a7 7 0 1 0-14 0m18 11l-6-6"/></symbol><use href="#ai:tabler:search"></use>  </svg> </a> </li> <li class="text-2xl"> <dark-light-toggle> <button title="Toggle dark & light mode" class="flex items-center"> <svg width="1em" height="1em" class="clickable block dark:hidden" data-icon="tabler:moon">   <symbol id="ai:tabler:moon" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 3h.393a7.5 7.5 0 0 0 7.92 12.446A9 9 0 1 1 12 2.992z"/></symbol><use href="#ai:tabler:moon"></use>  </svg> <svg width="1em" height="1em" class="clickable hidden dark:block" data-icon="tabler:sun">   <symbol id="ai:tabler:sun" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 12a4 4 0 1 0 8 0a4 4 0 1 0-8 0m-5 0h1m8-9v1m8 8h1m-9 8v1M5.6 5.6l.7.7m12.1-.7l-.7.7m0 11.4l.7.7m-12.1-.7l-.7.7"/></symbol><use href="#ai:tabler:sun"></use>  </svg> </button> </dark-light-toggle> <script type="module">class a extends HTMLElement{connectedCallback(){const e=this.querySelector("button");if(!e)return;const l=window.matchMedia("(prefers-color-scheme: dark)").matches,o=window.localStorage.getItem("theme")||(l?"dark":"light");window.mode.setMode(o),e.setAttribute("aria-label",o);const d=()=>{const t=window.mode.getMode()==="light"?"dark":"light";window.mode.setMode(t),window.localStorage.setItem("theme",t),e.setAttribute("aria-label",t)};e.addEventListener("click",d)}}customElements.get("dark-light-toggle")||customElements.define("dark-light-toggle",a);</script> </li> </ul> </nav> <div class="flex items-center justify-center gap-4 text-2xl sm:hidden"> <dark-light-toggle> <button title="Toggle dark & light mode" class="flex items-center"> <svg width="1em" height="1em" viewBox="0 0 24 24" class="clickable block dark:hidden" data-icon="tabler:moon">   <use href="#ai:tabler:moon"></use>  </svg> <svg width="1em" height="1em" viewBox="0 0 24 24" class="clickable hidden dark:block" data-icon="tabler:sun">   <use href="#ai:tabler:sun"></use>  </svg> </button> </dark-light-toggle>  <a href="/search/" aria-label="Search"> <svg width="1em" height="1em" viewBox="0 0 24 24" class="clickable text-xl" data-icon="tabler:search">   <use href="#ai:tabler:search"></use>  </svg> </a> <mobile-nav-toggle> <button class="flex items-center" title="Toggle mobile nav" aria-label="closed"> <!-- Menu icon (shown when closed) --> <svg width="1em" height="1em" id="mobile-nav-icon-closed" class="block" data-icon="tabler:menu-2">   <symbol id="ai:tabler:menu-2" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16"/></symbol><use href="#ai:tabler:menu-2"></use>  </svg> <svg width="1em" height="1em" id="mobile-nav-icon-open" class="hidden" data-icon="tabler:x">   <symbol id="ai:tabler:x" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M18 6L6 18M6 6l12 12"/></symbol><use href="#ai:tabler:x"></use>  </svg> </button> </mobile-nav-toggle> <script type="module">class a extends HTMLElement{connectedCallback(){const t=document.querySelector("#mobile-nav"),e=this.querySelector("#mobile-nav-icon-closed"),s=this.querySelector("#mobile-nav-icon-open"),o=this.querySelector("button");if(!t||!e||!s||!o)return;const n=()=>!t.classList.contains("hidden"),l=()=>{t.classList.add("hidden"),e.classList.remove("hidden"),e.classList.add("block"),s.classList.remove("block"),s.classList.add("hidden"),o.setAttribute("aria-label","closed")},c=()=>{t.classList.remove("hidden"),e.classList.remove("block"),e.classList.add("hidden"),s.classList.remove("hidden"),s.classList.add("block"),o.setAttribute("aria-label","open")},i=()=>{n()?l():c()};o.addEventListener("click",i)}}customElements.get("mobile-nav-toggle")||customElements.define("mobile-nav-toggle",a);</script> </div> </div> <hr class="border-separate border-accent/75 undefined"> <div id="mobile-nav" class="hidden"> <nav class="flex-col justify-end py-6"> <ul class="flex flex-col items-center gap-2"> <li class="flex flex-col justify-center"> <a href="/blog" class="flex items-center gap-1 whitespace-nowrap text-accent/100" target="_self">  <span class="underline"> Blog </span> </a> </li> </ul> </nav> <hr class="border-separate border-accent/75 undefined"> </div> </header> <div class="mt-4"> <a href="/blog/" class="clickable flex w-fit items-center gap-1"> <svg width="1em" height="1em" class="text-xl" data-icon="tabler:arrow-left">   <symbol id="ai:tabler:arrow-left" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 12h14M5 12l6 6m-6-6l6-6"/></symbol><use href="#ai:tabler:arrow-left"></use>  </svg>
  36. Back to all posts
  37. </a> </div> <aside class="scrollbar-hide fixed h-screen w-full max-w-sm -translate-x-full overflow-y-auto overscroll-y-none pr-8"> <div class="border-accent mt-24 hidden border-r pr-4 opacity-75 transition duration-500 ease-in-out hover:opacity-100 2xl:block
  38.         max-h-[calc(100vh-6rem)] overflow-y-auto sticky top-24"> <img src="https://cdn.blogweb.me/1/6555113c41b85bf4eb4c153f_d9e8cfc84a.png" alt="How to Make Collapsible Or Expandable Views In Swift? image" loading="lazy" decoding="async" fetchpriority="auto" width="200" height="100" class="mx-auto mb-6 max-w-[200px] w-full h-auto rounded-lg object-cover"> <div class="overflow-x-hidden p-4"> <div class="prose prose-lg max-w-full dark:prose-invert prose-h1:text-4xl prose-code:before:!content-none prose-code:after:!content-none sm:prose-h1:text-5xl prose-em:pr-[2px] undefined">  <h2 class="no-anchor mb-2">Table of Contents</h2> <details open> <summary class="cursor-pointer">Show more</summary> <ul class="mt-1 mb-1 list-none">
  39.          <li class="mt-1 mb-1">
  40.                              <a href="#how-to-handle-dynamic-content-in-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="How to handle dynamic content in collapsible views in Swift?">
  41.                                How to handle dynamic content in collapsible views in Swift?
  42.                              </a>
  43.                              
  44.                            </li><li class="mt-1 mb-1">
  45.                              <a href="#what-are-the-advantages-of-using-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="What are the advantages of using collapsible views in Swift?">
  46.                                What are the advantages of using collapsible views in Swift?
  47.                              </a>
  48.                              
  49.                            </li><li class="mt-1 mb-1">
  50.                              <a href="#what-is-the-impact-of-collapsible-views-on-app-performance-in-swift" class="no-underline hover:underline" aria-label="What is the impact of collapsible views on app performance in Swift?">
  51.                                What is the impact of collapsible views on app performance in Swift?
  52.                              </a>
  53.                              
  54.                            </li><li class="mt-1 mb-1">
  55.                              <a href="#what-is-the-recommended-way-to-structure-code-for-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="What is the recommended way to structure code for collapsible views in Swift?">
  56.                                What is the recommended way to structure code for collapsible views in Swift?
  57.                              </a>
  58.                              
  59.                            </li><li class="mt-1 mb-1">
  60.                              <a href="#how-to-add-headers-or-titles-to-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="How to add headers or titles to collapsible views in Swift?">
  61.                                How to add headers or titles to collapsible views in Swift?
  62.                              </a>
  63.                              
  64.                            </li>
  65.          </ul> </details>  </div>  </div> </div> </aside> <main id="main" class="mt-6 sm:mt-12"> <div class="prose prose-lg max-w-full dark:prose-invert prose-h1:text-4xl prose-code:before:!content-none prose-code:after:!content-none sm:prose-h1:text-5xl prose-em:pr-[2px] mb-6 sm:mb-12">    <h1 class="mt-2 mb-2 sm:my-4">How to Make Collapsible Or Expandable Views In Swift?</h1> <div class="flex gap-6 text-base">  </div> <div class="mt-4 flex justify-between"> <span class="opacity-75">Published on <time datetime="2025-09-20T16:37:19.174Z">Sep 20, 2025</time></span> <div class="flex gap-x-2 items-center"> <svg width="1em" height="1em" class="text-base" data-icon="tabler:clock">   <symbol id="ai:tabler:clock" viewBox="0 0 24 24"><g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"><path d="M3 12a9 9 0 1 0 18 0a9 9 0 0 0-18 0"/><path d="M12 7v5l3 3"/></g></symbol><use href="#ai:tabler:clock"></use>  </svg> <span class="whitespace-nowrap opacity-75">10 min read </span> </div> </div>  <div class="mt-8 w-full rounded-sm border 2xl:hidden"> <div class="overflow-x-hidden p-4"> <div class="prose prose-lg max-w-full dark:prose-invert prose-h1:text-4xl prose-code:before:!content-none prose-code:after:!content-none sm:prose-h1:text-5xl prose-em:pr-[2px] undefined">  <h2 class="no-anchor mb-2">Table of Contents</h2> <details> <summary class="cursor-pointer">Show more</summary> <ul class="mt-1 mb-1 list-none">
  66.          <li class="mt-1 mb-1">
  67.                              <a href="#how-to-handle-dynamic-content-in-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="How to handle dynamic content in collapsible views in Swift?">
  68.                                How to handle dynamic content in collapsible views in Swift?
  69.                              </a>
  70.                              
  71.                            </li><li class="mt-1 mb-1">
  72.                              <a href="#what-are-the-advantages-of-using-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="What are the advantages of using collapsible views in Swift?">
  73.                                What are the advantages of using collapsible views in Swift?
  74.                              </a>
  75.                              
  76.                            </li><li class="mt-1 mb-1">
  77.                              <a href="#what-is-the-impact-of-collapsible-views-on-app-performance-in-swift" class="no-underline hover:underline" aria-label="What is the impact of collapsible views on app performance in Swift?">
  78.                                What is the impact of collapsible views on app performance in Swift?
  79.                              </a>
  80.                              
  81.                            </li><li class="mt-1 mb-1">
  82.                              <a href="#what-is-the-recommended-way-to-structure-code-for-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="What is the recommended way to structure code for collapsible views in Swift?">
  83.                                What is the recommended way to structure code for collapsible views in Swift?
  84.                              </a>
  85.                              
  86.                            </li><li class="mt-1 mb-1">
  87.                              <a href="#how-to-add-headers-or-titles-to-collapsible-views-in-swift" class="no-underline hover:underline" aria-label="How to add headers or titles to collapsible views in Swift?">
  88.                                How to add headers or titles to collapsible views in Swift?
  89.                              </a>
  90.                              
  91.                            </li>
  92.          </ul> </details>  </div>  </div> </div> <div class="2xl:hidden"> <img src="https://cdn.blogweb.me/1/6555113c41b85bf4eb4c153f_d9e8cfc84a.png" alt="How to Make Collapsible Or Expandable Views In Swift? image" loading="lazy" decoding="async" fetchpriority="auto" width="400" height="300" class="mx-auto mb-6 max-w-[400px] w-full h-auto rounded-lg object-cover"> </div> <article data-pagefind-body>  <div class="max-w-7xl mx-auto"> <h2 class="ignore text-2xl font-bold mb-4 text-center">Best Swift UI Components to Buy in October 2025</h2> <div class="flex flex-col gap-6">  <style>astro-island,astro-slot,astro-static-slot{display:contents}</style><script>(()=>{var l=(n,t)=>{let i=async()=>{await(await n())()},e=typeof t.value=="object"?t.value:void 0,s={timeout:e==null?void 0:e.timeout};"requestIdleCallback"in window?window.requestIdleCallback(i,s):setTimeout(i,s.timeout||200)};(self.Astro||(self.Astro={})).idle=l;window.dispatchEvent(new Event("astro:idle"));})();</script><script>(()=>{var A=Object.defineProperty;var g=(i,o,a)=>o in i?A(i,o,{enumerable:!0,configurable:!0,writable:!0,value:a}):i[o]=a;var d=(i,o,a)=>g(i,typeof o!="symbol"?o+"":o,a);{let i={0:t=>m(t),1:t=>a(t),2:t=>new RegExp(t),3:t=>new Date(t),4:t=>new Map(a(t)),5:t=>new Set(a(t)),6:t=>BigInt(t),7:t=>new URL(t),8:t=>new Uint8Array(t),9:t=>new Uint16Array(t),10:t=>new Uint32Array(t),11:t=>1/0*t},o=t=>{let[l,e]=t;return l in i?i[l](e):void 0},a=t=>t.map(o),m=t=>typeof t!="object"||t===null?t:Object.fromEntries(Object.entries(t).map(([l,e])=>[l,o(e)]));class y extends HTMLElement{constructor(){super(...arguments);d(this,"Component");d(this,"hydrator");d(this,"hydrate",async()=>{var b;if(!this.hydrator||!this.isConnected)return;let e=(b=this.parentElement)==null?void 0:b.closest("astro-island[ssr]");if(e){e.addEventListener("astro:hydrate",this.hydrate,{once:!0});return}let c=this.querySelectorAll("astro-slot"),n={},h=this.querySelectorAll("template[data-astro-template]");for(let r of h){let s=r.closest(this.tagName);s!=null&&s.isSameNode(this)&&(n[r.getAttribute("data-astro-template")||"default"]=r.innerHTML,r.remove())}for(let r of c){let s=r.closest(this.tagName);s!=null&&s.isSameNode(this)&&(n[r.getAttribute("name")||"default"]=r.innerHTML)}let p;try{p=this.hasAttribute("props")?m(JSON.parse(this.getAttribute("props"))):{}}catch(r){let s=this.getAttribute("component-url")||"<unknown>",v=this.getAttribute("component-export");throw v&&(s+=` (export ${v})`),console.error(`[hydrate] Error parsing props for component ${s}`,this.getAttribute("props"),r),r}let u;await this.hydrator(this)(this.Component,p,n,{client:this.getAttribute("client")}),this.removeAttribute("ssr"),this.dispatchEvent(new CustomEvent("astro:hydrate"))});d(this,"unmount",()=>{this.isConnected||this.dispatchEvent(new CustomEvent("astro:unmount"))})}disconnectedCallback(){document.removeEventListener("astro:after-swap",this.unmount),document.addEventListener("astro:after-swap",this.unmount,{once:!0})}connectedCallback(){if(!this.hasAttribute("await-children")||document.readyState==="interactive"||document.readyState==="complete")this.childrenConnectedCallback();else{let e=()=>{document.removeEventListener("DOMContentLoaded",e),c.disconnect(),this.childrenConnectedCallback()},c=new MutationObserver(()=>{var n;((n=this.lastChild)==null?void 0:n.nodeType)===Node.COMMENT_NODE&&this.lastChild.nodeValue==="astro:end"&&(this.lastChild.remove(),e())});c.observe(this,{childList:!0}),document.addEventListener("DOMContentLoaded",e)}}async childrenConnectedCallback(){let e=this.getAttribute("before-hydration-url");e&&await import(e),this.start()}async start(){let e=JSON.parse(this.getAttribute("opts")),c=this.getAttribute("client");if(Astro[c]===void 0){window.addEventListener(`astro:${c}`,()=>this.start(),{once:!0});return}try{await Astro[c](async()=>{let n=this.getAttribute("renderer-url"),[h,{default:p}]=await Promise.all([import(this.getAttribute("component-url")),n?import(n):()=>()=>{}]),u=this.getAttribute("component-export")||"default";if(!u.includes("."))this.Component=h[u];else{this.Component=h;for(let f of u.split("."))this.Component=this.Component[f]}return this.hydrator=p,this.hydrate},e,this)}catch(n){console.error(`[astro-island] Error hydrating ${this.getAttribute("component-url")}`,n)}}attributeChangedCallback(){this.hydrate()}}d(y,"observedAttributes",["props"]),customElements.get("astro-island")||customElements.define("astro-island",y)}})();</script><astro-island uid="ZIw6ij" prefix="r1" component-url="/_astro/AddProductModal.DwKQYgIk.js" component-export="default" renderer-url="/_astro/client.BPIbHqJh.js" props="{&quot;isProd&quot;:[0,true],&quot;sponsoredCost&quot;:[0,15]}" ssr client="idle" opts="{&quot;name&quot;:&quot;AddProductModal&quot;,&quot;value&quot;:true}" await-children><div class="flex md:flex-row flex-col cursor-pointer w-full border rounded-2xl p-4 shadow-lg flex items-center gap-4 border-dashed border-gray-300 dark:border-gray-600 bg-gray-50 dark:bg-gray-800 relative"><span class="absolute top-2 left-2 text-lg font-bold border border-2 border-gray-400 w-10 h-10 flex items-center justify-center rounded-full text-gray-400">+</span><div class="w-[100px] h-[100px] mt-10 bg-gray-300 dark:bg-gray-700 rounded-lg"></div><div class="flex-1 space-y-2 min-h-[80px] w-full"><div class="h-5 bg-gray-300 dark:bg-gray-700 mb-5 rounded w-3/4"></div><div class="h-4 bg-gray-200 dark:bg-gray-600 rounded w-2/4"></div><div class="h-4 bg-gray-200 dark:bg-gray-600 rounded w-2/4"></div><div class="h-4 bg-gray-200 dark:bg-gray-600 rounded w-2/4"></div></div><div class="flex flex-col items-center space-y-2"><span class="bg-orange-500 text-white dark:text-black font-bold text-center py-2 px-4 rounded-xl hover:bg-orange-600 transition">ONE MORE?</span></div></div><!--astro:end--></astro-island> </div> </div> <p>In Swift, you can create collapsible or expandable views using a variety of techniques. One commonly used approach involves using a combination of views, constraints, and animations.</p>
  93. <p>Here's a step-by-step guide on how to achieve collapsible or expandable views in Swift:</p>
  94. <ol>
  95. <li><strong>Create two views</strong>: one represents the collapsed state, and the other represents the expanded state of your <a href="https://ubuntuask.com/blog/how-to-get-the-actual-height-of-content-view-in">content</a>. These can be UIImageViews, UILabels, or any other type of view you want to animate.</li>
  96. <li>Set up appropriate constraints for both views so that they stack on top of each other. For example, you can use vertical spacing constraints to ensure they are aligned.</li>
  97. <li>Set up an IBOutlet for the constraint that determines the height of the expanded view. This way, you can easily modify its value later.</li>
  98. <li>Create a function that toggles the state of the collapsible view. This function should animate the change in height and any other desired animations. One way to achieve this is by updating the constant value of the constraint in an animation block and calling <a href="https://studentprojectcode.com/blog/how-to-use-css3-flexbox-for-layout">layout</a>IfNeeded() to trigger the layout update. For example: @IBAction func toggleView(_ sender: UIButton) { UIView.animate(withDuration: 0.3) { self.expandedViewHeightConstraint.constant = self.collapsedView.isHidden ? self.expandedViewHeight : 0 self.view.layoutIfNeeded() } }</li>
  99. <li>Hook up this toggle function to a user action, such as a button tap or a swipe gesture recognizer.</li>
  100. </ol>
  101. <p>By following these steps, you can create collapsible or expandable views in Swift. Remember to customize the animations, views, and constraints as per your specific needs.</p>
  102. <h2 id="user-content-how-to-handle-dynamic-content-in-collapsible-views-in-swift">How to handle dynamic content in collapsible views in Swift?</h2>
  103. <p>To handle dynamic content in collapsible views in Swift, you can use a combination of UITableView and UIStackView. Here are the steps to achieve this:</p>
  104. <ol>
  105. <li>Create a UITableView with a custom UITableViewCell that contains a UIStackView as its contentView. This stack view will hold the dynamic content.</li>
  106. <li>Configure your UITableView to have collapsible sections. For example, you can use an array to keep track of the expanded/collapsed state for each section. When a header of a section is tapped, you can update the state and reload the corresponding section.</li>
  107. <li>Create a function in your custom UITableViewCell to set the dynamic content. This function takes an array of data and creates the necessary UI components (e.g., UILabels, UIImageViews) dynamically within the stack view.</li>
  108. <li>When a section is expanded or collapsed, update the UIStackView's arrangedSubviews property to add or remove the necessary dynamic UI components. You can also update the constraints of the UIStackView to accommodate the new content.</li>
  109. </ol>
  110. <p>Here is some sample code to help you get started:</p>
  111. <p>// Custom UITableViewCell containing a UIStackView
  112. class CollapsibleTableViewCell: UITableViewCell {
  113. @IBOutlet weak var stackView: UIStackView!</p>
  114. <pre><code>func setDynamicContent(content: \[String\]) {
  115.    // Remove previous dynamic content
  116.    stackView.arrangedSubviews.forEach {
  117.        stackView.removeArrangedSubview($0)
  118.        $0.removeFromSuperview()
  119.    }
  120.    
  121.    // Add new dynamic content
  122.    for item in content {
  123.        let label = UILabel()
  124.        label.text = item
  125.        stackView.addArrangedSubview(label)
  126.    }
  127. }
  128. </code></pre>
  129. <p>}</p>
  130. <p>// Your UITableViewController
  131. class CollapsibleTableViewController: UITableViewController {
  132. var sectionsExpanded = [Bool]()
  133. var contentArray = [[String]]()</p>
  134. <pre><code>override func viewDidLoad() {
  135.    super.viewDidLoad()
  136.    // Initialize sectionsExpanded and contentArray with your data
  137. }
  138.  
  139. override func numberOfSections(in tableView: UITableView) -> Int {
  140.    return sectionsExpanded.count
  141. }
  142.  
  143. override func tableView(\_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  144.    return sectionsExpanded\[section\] ? contentArray\[section\].count : 0
  145. }
  146.  
  147. override func tableView(\_ tableView: UITableView, viewForHeaderInSection section: Int) -> UIView? {
  148.    // Configure and return the header view
  149. }
  150.  
  151. override func tableView(\_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  152.    let cell = tableView.dequeueReusableCell(withIdentifier: "CollapsibleCell", for: indexPath) as! CollapsibleTableViewCell
  153.    cell.setDynamicContent(content: contentArray\[indexPath.section\])
  154.    return cell
  155. }
  156.  
  157. override func tableView(\_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  158.    sectionsExpanded\[indexPath.section\].toggle()
  159.    tableView.reloadSections(\[indexPath.section\], with: .fade)
  160. }
  161. </code></pre>
  162. <p>}</p>
  163. <p>Here, each section in the table view represents a collapsible view, and the dynamic content is updated based on the expanded/collapsed state. Remember to customize and adapt the code to fit your specific requirements.</p>
  164. <h2 id="user-content-what-are-the-advantages-of-using-collapsible-views-in-swift">What are the advantages of using collapsible views in Swift?</h2>
  165. <p>There are several advantages of using collapsible views in Swift:</p>
  166. <ol>
  167. <li><strong>Space-saving</strong>: Collapsible views allow you to hide content when it is not needed, which saves screen space and improves the overall user experience. This is particularly useful in situations where there is limited space or when you want to provide the ability to focus on specific content.</li>
  168. <li><strong>Improved navigation</strong>: Collapsible views can help in managing complex <a href="https://studentprojectcode.com/blog/how-to-use-swiftui-for-building-user-interfaces-in">user interfaces</a> by allowing users to expand or collapse sections of content as needed. This makes it easier for users to navigate and find the information they need without overwhelming them with too much information at once.</li>
  169. <li><strong>Enhancing user interaction</strong>: Collapsible views can provide interactive and dynamic user experiences. For example, you can collapse or expand sections based on user interactions, such as tapping a button or swiping a cell. This can make the application feel more engaging and responsive to user actions.</li>
  170. <li><strong>Organizational structure</strong>: Collapsible views can be useful for organizing content hierarchically. By grouping related information together and allowing users to collapse or expand them, you can improve the clarity and organization of your user interface.</li>
  171. <li><strong>Customizability</strong>: Collapsible views can be customized to match the visual style and aesthetics of your application. You can design the appearance, animations, and transition effects of expand and collapse actions to create a unique and visually appealing user interface.</li>
  172. </ol>
  173. <p>Overall, collapsible views provide a flexible and effective way to manage content and enhance user interaction, making your application more intuitive and user-friendly.</p>
  174. <h2 id="user-content-what-is-the-impact-of-collapsible-views-on-app-performance-in-swift">What is the impact of collapsible views on app performance in Swift?</h2>
  175. <p>Collapsible views can have a significant impact on app performance if not implemented properly. Here are a few factors to consider:</p>
  176. <ol>
  177. <li><strong>Memory Usage</strong>: When views are collapsed, the memory occupied by their contents is not released. If you have many collapsible views with heavy content, it can lead to increased memory consumption, potentially causing memory warnings or even app crashes.</li>
  178. <li><strong>Layout and Rendering</strong>: Expanding or collapsing views often requires modifying the layout of the surrounding elements. Frequent layout changes may result in frequent re-rendering, leading to decreased app performance, especially if your view hierarchy is complex or contains heavy elements.</li>
  179. <li><strong>Animations</strong>: Implementing animations during collapsible view transitions can have an impact on performance. If the animations are not optimized or if too many views are animated simultaneously, it can result in choppy animations and reduced overall app responsiveness.</li>
  180. </ol>
  181. <p>To mitigate these performance impacts, here are some best practices:</p>
  182. <ol>
  183. <li><strong>Lazy Loading</strong>: Load the content of collapsible views only when necessary, i.e., when the view is expanded. This way, you can reduce memory usage by loading content only when it is needed.</li>
  184. <li><strong>Reuse Cells/Views</strong>: If you are using a table view or collection view to implement collapsible views, make sure to properly reuse cells/views to minimize the memory footprint.</li>
  185. <li><strong>Optimize Animations</strong>: Use efficient animation techniques like Core Animation, UIViewPropertyAnimator, or UIKit Dynamics to ensure smooth and performant animations.</li>
  186. <li><strong>Profile and Optimize</strong>: Use profiling tools like Instruments to identify performance bottlenecks and optimize your code. For example, you can identify unnecessary layout recalculations or memory leaks that can impact performance.</li>
  187. </ol>
  188. <p>By carefully considering these factors and adopting appropriate optimizations, you can ensure that collapsible views have a minimal impact on your app's performance in Swift.</p>
  189. <h2 id="user-content-what-is-the-recommended-way-to-structure-code-for-collapsible-views-in-swift">What is the recommended way to structure code for collapsible views in Swift?</h2>
  190. <p>There are multiple ways to structure code for collapsible views in Swift, depending on the complexity and requirements of your app. Here is one recommended approach:</p>
  191. <ol>
  192. <li>Create a custom view or UIView subclass that represents the collapsible view. This view should contain the necessary UI elements and logic to handle the collapsed and expanded states.</li>
  193. <li>Create a delegate protocol for the collapsible view. This protocol should define methods that allow the parent view or view controller to respond to events and changes in the collapsible view's state.</li>
  194. <li>Implement the collapsible view logic in the custom view or UIView subclass. This includes handling user interactions, updating UI elements, and managing the collapsed and expanded states of the view.</li>
  195. <li>Use auto layout or other layout mechanisms to position and size the collapsible view within the parent view or view controller.</li>
  196. <li>Implement the delegate protocol methods in the parent view or view controller to handle events and changes from the collapsible view.</li>
  197. <li>Update the UI and state of the parent view or view controller based on the changes received from the collapsible view.</li>
  198. </ol>
  199. <p>By following this structure, you can encapsulate the code and functionality related to the collapsible view in a separate module, making it easier to maintain and reuse in different parts of your app.</p>
  200. <h2 id="user-content-how-to-add-headers-or-titles-to-collapsible-views-in-swift">How to add headers or titles to collapsible views in Swift?</h2>
  201. <p>To add headers or titles to collapsible views in Swift, you can use section headers in a UITableView or a UICollectionReusableView in a UICollectionView. Here are the steps to do so:</p>
  202. <ol>
  203. <li>Create a custom view for the header that will act as the collapsible view's title. You can do this by subclassing UITableViewHeaderFooterView for UITableView or UICollectionReusableView for UICollectionView.</li>
  204. <li>Add any necessary labels, buttons, or other UI elements to the custom header view to display the title or any other relevant information.</li>
  205. <li>Override the initializer for the custom header view to set up the UI elements in the view. For example, if using a UILabel to display the title, you can initialize and customize it in this step.</li>
  206. <li><strong>In your UITableView or UICollectionView data source, implement the method tableView(_</strong>:viewForHeaderInSection:) for UITableView or collectionView(_:viewForSupplementaryElementOfKind:at:) for UICollectionView. In these methods, you can dequeue the custom header view and configure it with the required data.</li>
  207. <li>Return the custom header view from the above method, and it will be displayed as the title for the collapsible view.</li>
  208. </ol>
  209. <p>Note: For collapsible sections, you can implement the UITableViewDataSource methods in UITableView or UICollectionViewDataSource methods in UICollectionViewDataSource to control the numbers of rows/items, section headers, and section footers.</p>
  210. <p>Here is an example implementation for a UITableView:</p>
  211. <p>// CustomHeaderView.swift
  212. import UIKit</p>
  213. <p>class CustomHeaderView: UITableViewHeaderFooterView {
  214. let titleLabel = UILabel()</p>
  215. <pre><code>override init(reuseIdentifier: String?) {
  216.    super.init(reuseIdentifier: reuseIdentifier)
  217.    // Customize the header view
  218.    titleLabel.textColor = .black
  219.    titleLabel.font = UIFont.boldSystemFont(ofSize: 16)
  220.    contentView.addSubview(titleLabel)
  221.    
  222.    // Set up constraints for the titleLabel
  223.    titleLabel.translatesAutoresizingMaskIntoConstraints = false
  224.    titleLabel.leadingAnchor.constraint(equalTo: contentView.leadingAnchor, constant: 16).isActive = true
  225.    titleLabel.topAnchor.constraint(equalTo: contentView.topAnchor, constant: 8).isActive = true
  226.    titleLabel.trailingAnchor.constraint(equalTo: contentView.trailingAnchor, constant: -16).isActive = true
  227.    titleLabel.bottomAnchor.constraint(equalTo: contentView.bottomAnchor, constant: -8).isActive = true
  228. }
  229.  
  230. required init?(coder aDecoder: NSCoder) {
  231.    fatalError("init(coder:) has not been implemented")
  232. }
  233. </code></pre>
  234. <p>}</p>
  235. <p>// ViewController.swift
  236. import UIKit</p>
  237. <p>class ViewController: UIViewController, UITableViewDataSource {
  238. let tableView = UITableView()</p>
  239. <pre><code>override func viewDidLoad() {
  240.    super.viewDidLoad()
  241.    
  242.    tableView.register(CustomHeaderView.self, forHeaderFooterViewReuseIdentifier: "CustomHeader")
  243.    tableView.dataSource = self
  244.    
  245.    // Set up constraints for the tableView
  246.    tableView.translatesAutoresizingMaskIntoConstraints = false
  247.    tableView.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
  248.    tableView.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
  249.    tableView.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
  250.    tableView.bottomAnchor.constraint(equalTo: view.bottomAnchor).isActive = true
  251.    
  252.    view.addSubview(tableView)
  253. }
  254.  
  255. func numberOfSections(in tableView: UITableView) -> Int {
  256.    return 1
  257. }
  258.  
  259. func tableView(\_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  260.    return 5
  261. }
  262.  
  263. func tableView(\_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  264.    let cell = UITableViewCell(style: .default, reuseIdentifier: "Cell")
  265.    cell.textLabel?.text = "Row \\(indexPath.row + 1)"
  266.    return cell
  267. }
  268.  
  269. func tableView(\_ tableView: UITableView, viewForHeaderInSection section: Int) -> UIView? {
  270.    let headerView = tableView.dequeueReusableHeaderFooterView(withIdentifier: "CustomHeader") as? CustomHeaderView
  271.    headerView?.titleLabel.text = "Section 1"
  272.    return headerView
  273. }
  274. // If you want to make the header view collapsible, you can add a tap gesture recognizer to the header and handle the tap action accordingly.
  275. </code></pre>
  276. <p>}</p>
  277. <p>This example creates a custom <code>UITableViewHeaderFooterView</code> subclass called <code>CustomHeaderView</code>. The <code>CustomHeaderView</code> has a <code>UILabel</code> to display the title. The <code>titleLabel</code> is added as a subview to the <code>contentView</code> of the header view.</p>
  278. <p>In <code>viewForHeaderInSection</code> of the <code>UITableViewDataSource</code>, the custom header view is dequeued and configured with the appropriate title "Section 1". Finally, the custom header view is returned to be displayed as the title for the collapsible section.</p>
  279. </article> <div class="mt-5 flex justify-center"> <script>(()=>{var e=async t=>{await(await t())()};(self.Astro||(self.Astro={})).only=e;window.dispatchEvent(new Event("astro:only"));})();</script><astro-island uid="1hmnHg" component-url="/_astro/ShareInSocialNetworks.AWxQj_p-.js" component-export="default" renderer-url="/_astro/client.BPIbHqJh.js" props="{&quot;url&quot;:[0,&quot;https://studentprojectcode.com/blog/how-to-make-collapsible-or-expandable-views-in.html&quot;],&quot;title&quot;:[0,&quot;How to Make Collapsible Or Expandable Views In Swift?&quot;]}" ssr client="only" opts="{&quot;name&quot;:&quot;ShareInSocialNetworks&quot;,&quot;value&quot;:&quot;react&quot;}"></astro-island> </div>  <script data-astro-rerun>
  280.    function setupHeadingAnchors() {
  281.        const headings = Array.from(document.querySelectorAll('h2, h3, h4, h5, h6'))
  282.            .filter(heading => !heading.classList.contains('ignore'));
  283.  
  284.        for (const heading of headings) {
  285.            if (heading.classList.contains('no-anchor')) continue;
  286.  
  287.            heading.classList.add('group');
  288.            heading.id = heading.innerText.trim().toLowerCase().replace(/\s+/g, '-').replace(/[^\w-]/g, '');
  289.  
  290.            const link = document.createElement('a');
  291.            link.innerText = '#';
  292.            link.className = 'heading-link lg:hidden inline group-hover:inline-block ml-3 no-underline';
  293.            link.href = '#' + heading.id;
  294.            link.setAttribute('aria-hidden', 'true');
  295.  
  296.            heading.appendChild(link);
  297.        }
  298.    }
  299.  
  300.    setupHeadingAnchors();
  301. </script>   </div>  </main> <div class="flex grow flex-col justify-end"> <div class="my-4 grid grid-cols-3"> <div class="flex flex-col items-start"> <a href="/blog/how-to-quickly-deploy-gatsby-on-linode" class="clickable"> <div class="flex items-center justify-start gap-1"> <svg width="1em" height="1em" viewBox="0 0 24 24" class="text-xl" data-icon="tabler:arrow-left">   <use href="#ai:tabler:arrow-left"></use>  </svg> <span>Next post</span> </div> <span class="mt-2 text-lg sm:text-xl">How to Quickly Deploy Gatsby on Linode?</span> </a> </div> <div class="col-start-3 flex flex-col items-end"> <a href="/blog/tutorial-run-next-js-on-dreamhost" class="clickable"> <div class="flex items-center justify-end gap-1"> <span>Previous post</span> <svg width="1em" height="1em" class="text-xl" data-icon="tabler:arrow-right">   <symbol id="ai:tabler:arrow-right" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 12h14m-6 6l6-6m-6-6l6 6"/></symbol><use href="#ai:tabler:arrow-right"></use>  </svg> </div> <span class="mt-2 text-lg sm:text-xl">Tutorial: Run Next.js on DreamHost?</span> </a> </div> </div> </div> <footer class="mt-auto"> <hr class="border-separate border-accent/75 undefined"> <div class="flex items-center flex-col md:flex-row justify-between gap-2 py-4 text-2xl sm:py-8"> <span class="text-base">&#169; 2025 Copyright: studentprojectcode.com</span> <div class="grow"></div> <div class="flex flex-col items-center justify-center gap-y-2 sm:flex-row sm:gap-x-4 sm:gap-y-0"> <a class="text-base text-center" href="/terms/" aria-label="Terms" rel="noopener">
  302. Terms
  303. </a> <a class="text-base text-center" href="/privacy-policy/" aria-label="Privacy Policy" rel="noopener">
  304. Privacy
  305. </a> <a href="#" data-cc="show-preferencesModal" class="text-base text-center">
  306. Your privacy choices
  307. </a> <a class="text-base text-center" href="/contact/" aria-label="Contact us" rel="noopener">
  308. Contact us
  309. </a> </div> <a class="flex flex-col justify-center" href="https://x.com/JordanKEurope" target="_blank" aria-label="Twitter" rel="noopener noreferrer"> <svg width="1em" height="1em" class="clickable" data-icon="tabler:brand-x"> <title>Twitter</title>  <symbol id="ai:tabler:brand-x" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m4 4l11.733 16H20L8.267 4zm0 16l6.768-6.768m2.46-2.46L20 4"/></symbol><use href="#ai:tabler:brand-x"></use>  </svg> </a> </div> </footer> </div> <scroll-progress-bar class="fixed top-0 w-full z-50"> <div class="h-1 w-0 bg-gradient-to-r from-orange-500 via-yellow-500 to-pink-500 transition-all duration-150 ease-linear"></div> </scroll-progress-bar> <script type="module">class o extends HTMLElement{connectedCallback(){const t=this.children[0];function c(){const e=document.body.scrollTop||document.documentElement.scrollTop,l=document.documentElement.scrollHeight-document.documentElement.clientHeight;return e/l*100}document.addEventListener("scroll",()=>{const e=c();t.style.width=`${e}%`})}}customElements.get("scroll-progress-bar")||customElements.define("scroll-progress-bar",o);</script><scroll-top-button class="fixed! right-4 bottom-4 z-10 hidden pr-2 sm:right-8 sm:bottom-8 lg:right-16 lg:bottom-16"> <button class="clickable flex items-center gap-2" title="Scroll to top" aria-label="Scroll to top"> <svg width="1em" height="1em" aria-hidden="true" class="text-4xl" data-icon="tabler:arrow-up">   <symbol id="ai:tabler:arrow-up" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 5v14m6-8l-6-6m-6 6l6-6"/></symbol><use href="#ai:tabler:arrow-up"></use>  </svg> </button> </scroll-top-button> <script type="module">class n extends HTMLElement{connectedCallback(){const t=this.querySelector("button");if(!t)return;function e(){const o=document.documentElement.clientHeight;return(document.body.scrollTop||document.documentElement.scrollTop)>o}t.addEventListener("click",()=>{document.body.scrollTop=0,document.documentElement.scrollTop=0}),document.addEventListener("scroll",()=>{e()?this.classList.remove("hidden"):this.classList.add("hidden")})}}customElements.get("scroll-top-button")||customElements.define("scroll-top-button",n);</script> <button class="copy-button" title="Copy"> <svg width="1em" height="1em" class="copy-btn hidden" data-icon="tabler--copy" data-icon="tabler:copy">   <symbol id="ai:tabler:copy" viewBox="0 0 24 24"><g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"><path d="M7 9.667A2.667 2.667 0 0 1 9.667 7h8.666A2.667 2.667 0 0 1 21 9.667v8.666A2.667 2.667 0 0 1 18.333 21H9.667A2.667 2.667 0 0 1 7 18.333z"/><path d="M4.012 16.737A2 2 0 0 1 3 15V5c0-1.1.9-2 2-2h10c.75 0 1.158.385 1.5 1"/></g></symbol><use href="#ai:tabler:copy"></use>  </svg> <svg width="1em" height="1em" class="copy-success hidden" data-icon="tabler--check" data-icon="tabler:check">   <symbol id="ai:tabler:check" viewBox="0 0 24 24"><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m5 12l5 5L20 7"/></symbol><use href="#ai:tabler:check"></use>  </svg> </button> <script data-astro-rerun>
  310.    function setupCopyCodeButtons() {
  311.        document.querySelectorAll('.astro-code-header > button').forEach((btn) => {
  312.            btn.addEventListener('click', async () => {
  313.                const code = btn.closest('.astro-code-header').nextElementSibling.querySelector('.astro-code code');
  314.                const copyIcon = btn.querySelector('.copy-btn');
  315.                const successIcon = btn.querySelector('.copy-success');
  316.  
  317.                if (!code || !copyIcon || !successIcon) return;
  318.  
  319.                await navigator.clipboard.writeText(code.innerText);
  320.  
  321.                // Toggle visibility
  322.                copyIcon.classList.add('hidden');
  323.                successIcon.classList.remove('hidden');
  324.  
  325.                // Revert after 1 second
  326.                setTimeout(() => {
  327.                    copyIcon.classList.remove('hidden');
  328.                    successIcon.classList.add('hidden');
  329.                    }, 1000);
  330.                });
  331.            });
  332.        }
  333.  
  334.        setupCopyCodeButtons();
  335. </script> <script data-astro-rerun src="/zoom-vanilla.js/zoom-vanilla.min.js"></script> <script defer src="https://static.cloudflareinsights.com/beacon.min.js/vcd15cbe7772f49c399c6a5babf22c1241717689176015" integrity="sha512-ZpsOmlRQV6y907TI0dKBHq9Md29nnaEIPlkf84rnaERnq6zvWvPUqr2ft8M1aS28oN72PdrCzSjY4U6VaAw1EQ==" data-cf-beacon='{"version":"2024.11.0","token":"954545f75d1041c7bd12c69bdbb908d6","r":1,"server_timing":{"name":{"cfCacheStatus":true,"cfEdge":true,"cfExtPri":true,"cfL4":true,"cfOrigin":true,"cfSpeedBrain":true},"location_startswith":null}}' crossorigin="anonymous"></script>
  336. </body> </html>
Copyright © 2002-9 Sam Ruby, Mark Pilgrim, Joseph Walton, and Phil Ringnalda