<?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: Kiran Balaji</title>
    <description>The latest articles on DEV Community by Kiran Balaji (@kiran_balaji_197).</description>
    <link>https://dev.to/kiran_balaji_197</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%2F2882592%2Fec125d30-1543-4780-865a-839b2833887a.png</url>
      <title>DEV Community: Kiran Balaji</title>
      <link>https://dev.to/kiran_balaji_197</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/kiran_balaji_197"/>
    <language>en</language>
    <item>
      <title>Building a Premium Bento-Style Portfolio with React, GSAP &amp; Tailwind v4</title>
      <dc:creator>Kiran Balaji</dc:creator>
      <pubDate>Mon, 29 Dec 2025 16:55:17 +0000</pubDate>
      <link>https://dev.to/kiran_balaji_197/building-a-premium-bento-style-portfolio-with-react-gsap-tailwind-v4-2ig8</link>
      <guid>https://dev.to/kiran_balaji_197/building-a-premium-bento-style-portfolio-with-react-gsap-tailwind-v4-2ig8</guid>
      <description>&lt;p&gt;Creating my portfolio is a rite of passage, but I wanted something that felt more like an OS dashboard than a static resume. After exploring modern design trends, I decided to build a &lt;strong&gt;Bento-style grid&lt;/strong&gt; that feels alive, responsive, and data-driven.&lt;/p&gt;

&lt;p&gt;Here’s a breakdown of how I built it and the tech stack behind it.&lt;/p&gt;

&lt;h2&gt;
  
  
  🎨 The Design: The "Bento" Philosophy
&lt;/h2&gt;

&lt;p&gt;The Bento grid (inspired by Apple’s promotional materials) allows for a high density of information without feeling cluttered. Each "card" has its own purpose:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Profile Card&lt;/strong&gt;: A quick "Hello".&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Dynamic Stats&lt;/strong&gt;: Live viewer counts and GitHub metrics.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Experience &amp;amp; Education&lt;/strong&gt;: Structured timelines.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Projects&lt;/strong&gt;: A sleek carousel to showcase work.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  🛠️ The Tech Stack
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Frontend&lt;/strong&gt;: React 19 + TypeScript (for type safety).&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Styling&lt;/strong&gt;: Tailwind CSS (leveraging the new v4 features for clean variable handling).&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Animations&lt;/strong&gt;: GSAP (GreenSock) for those "butter-smooth" entry transitions.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Icons&lt;/strong&gt;: Lucide-React (consistent and lightweight).&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Data Hooking&lt;/strong&gt;: SWR for efficient fetching from a custom API.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  🚀 Key Technical Features
&lt;/h2&gt;

&lt;h3&gt;
  
  
  1. The Entrance Animation (GSAP)
&lt;/h3&gt;

&lt;p&gt;Rather than just appearing, the cards stagger into view with a scale-up effect using GSAP's &lt;code&gt;ScrollTrigger&lt;/code&gt;. It gives a sense of "weight" to the UI as the user scrolls down.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight tsx"&gt;&lt;code&gt;&lt;span class="nx"&gt;gsap&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;fromTo&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;.bento-card&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;opacity&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;y&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;50&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;scale&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.9&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
  &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;opacity&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;1&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;y&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;scale&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;1&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;stagger&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.1&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;duration&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.8&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;ease&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;power3.out&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  2. Optimized Icon Visibility
&lt;/h3&gt;

&lt;p&gt;One challenge with dark mode is that logos like GitHub or Express are naturally black. I implemented a system that automatically detects these icons and applies a &lt;code&gt;dark:invert&lt;/code&gt; filter, ensuring they pop regardless of the user's theme.&lt;/p&gt;

&lt;h3&gt;
  
  
  3. The "Fullstack" Skills Card
&lt;/h3&gt;

&lt;p&gt;Instead of a simple list, I organized skills into categories (Frontend, Backend, Tools) with a "layered" UI. Each segment has:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;A bold category title with a horizontal separator.&lt;/li&gt;
&lt;li&gt;A subtle, large background watermark text for a "premium" depth effect.&lt;/li&gt;
&lt;li&gt;High-quality SVG icons for every technology.&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  4. Interactive Social Hub
&lt;/h3&gt;

&lt;p&gt;The social links aren't just buttons. They are interactive cards that support:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Instagram/YouTube/X&lt;/strong&gt; integration.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Direct Resume Download&lt;/strong&gt;: Using the &lt;code&gt;download&lt;/code&gt; attribute on a specialized PDF link.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Hover States&lt;/strong&gt;: Cards rotate and scale slightly on hover, making the grid feel interactive.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  📈 Live Data Integration
&lt;/h2&gt;

&lt;p&gt;I didn’t want static numbers. The portfolio connects to a custom API server that:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Fetches real-time &lt;strong&gt;GitHub Stats&lt;/strong&gt; (Stars, Repos, Contributions), &lt;strong&gt;GitHub Project Repos&lt;/strong&gt;.&lt;/li&gt;
&lt;li&gt;Tracks &lt;strong&gt;Unique Viewers&lt;/strong&gt; and &lt;strong&gt;Likes&lt;/strong&gt; using a visitor ID system stored in local storage.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://media2.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%2Flhe4ojn12843y9xjjjct.jpeg" class="article-body-image-wrapper"&gt;&lt;img src="https://media2.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%2Flhe4ojn12843y9xjjjct.jpeg" alt=" " width="800" height="1248"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h2&gt;
  
  
  🔗 Links
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Live Demo&lt;/strong&gt;: [&lt;a href="https://my-portfolio-two-phi-2q08hcotfn.vercel.app/" rel="noopener noreferrer"&gt;https://my-portfolio-two-phi-2q08hcotfn.vercel.app/&lt;/a&gt;]&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;Bento portfolio just dropped 👀&lt;br&gt;
Rate it -----/⭐⭐⭐⭐⭐ | roast it in the comments 🔥💬&lt;/strong&gt;&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>typescript</category>
      <category>react</category>
      <category>portfolio</category>
    </item>
    <item>
      <title>Controlled vs Uncontrolled Forms in React 🚀</title>
      <dc:creator>Kiran Balaji</dc:creator>
      <pubDate>Fri, 26 Dec 2025 16:01:02 +0000</pubDate>
      <link>https://dev.to/kiran_balaji_197/controlled-vs-uncontrolled-forms-in-react-219a</link>
      <guid>https://dev.to/kiran_balaji_197/controlled-vs-uncontrolled-forms-in-react-219a</guid>
      <description>&lt;p&gt;Forms look simple in React—until they aren’t.&lt;/p&gt;

&lt;p&gt;If you’ve ever wondered:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Why is my input re-rendering so much?&lt;/li&gt;
&lt;li&gt;When should I use useRef instead of useState?&lt;/li&gt;
&lt;li&gt;What do interviewers actually expect here?&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This article breaks down controlled vs uncontrolled forms in React with practical use cases, not just definitions.&lt;/p&gt;




&lt;h2&gt;
  
  
  What Is a Controlled Form?
&lt;/h2&gt;

&lt;p&gt;A controlled form is one where React controls the input value.&lt;br&gt;
The input’s value lives in React state, and every change goes through an onChange handler.&lt;/p&gt;

&lt;p&gt;Example: &lt;strong&gt;Controlled Input&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;import { useState } from "react";

function ControlledForm() {
  const [email, setEmail] = useState("");

  return (
    &amp;lt;form&amp;gt;
      &amp;lt;input
        type="email"
        value={email}
        onChange={(e) =&amp;gt; setEmail(e.target.value)}
        placeholder="Enter email"
      /&amp;gt;
      &amp;lt;p&amp;gt;Email: {email}&amp;lt;/p&amp;gt;
    &amp;lt;/form&amp;gt;
  );
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;✅ Pros of Controlled Forms&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Easy validation&lt;/li&gt;
&lt;li&gt;Instant UI updates&lt;/li&gt;
&lt;li&gt;Predictable behavior&lt;/li&gt;
&lt;li&gt;Required for complex logic&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;❌ Cons of Controlled Forms&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;More boilerplate&lt;/li&gt;
&lt;li&gt;Re-renders on every keystroke&lt;/li&gt;
&lt;li&gt;Can hurt performance in large forms&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  What Is an Uncontrolled Form?
&lt;/h2&gt;

&lt;p&gt;An uncontrolled form lets the DOM handle the input state.&lt;br&gt;
React accesses the value only when needed, using a ref.&lt;/p&gt;

&lt;p&gt;Example: &lt;strong&gt;Uncontrolled Input&lt;/strong&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;import { useRef } from "react";

function UncontrolledForm() {
  const emailRef = useRef&amp;lt;HTMLInputElement&amp;gt;(null);

  const handleSubmit = (e: React.FormEvent) =&amp;gt; {
    e.preventDefault();
    console.log(emailRef.current?.value);
  };

  return (
    &amp;lt;form onSubmit={handleSubmit}&amp;gt;
      &amp;lt;input type="email" ref={emailRef} placeholder="Enter email" /&amp;gt;
      &amp;lt;button type="submit"&amp;gt;Submit&amp;lt;/button&amp;gt;
    &amp;lt;/form&amp;gt;
  );
}
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;✅ Pros of Uncontrolled Forms&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Better performance&lt;/li&gt;
&lt;li&gt;Less code&lt;/li&gt;
&lt;li&gt;Closer to native HTML&lt;/li&gt;
&lt;li&gt;Great for simple forms&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;❌ Cons of Uncontrolled Forms&lt;/strong&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Harder validation&lt;/li&gt;
&lt;li&gt;No instant UI feedback&lt;/li&gt;
&lt;li&gt;Not ideal for dynamic logic&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Controlled vs Uncontrolled: Side-by-Side
&lt;/h2&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Feature&lt;/th&gt;
&lt;th&gt;Controlled&lt;/th&gt;
&lt;th&gt;Uncontrolled&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;State location&lt;/td&gt;
&lt;td&gt;React&lt;/td&gt;
&lt;td&gt;DOM&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Re-renders&lt;/td&gt;
&lt;td&gt;Every change&lt;/td&gt;
&lt;td&gt;On submit&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Validation&lt;/td&gt;
&lt;td&gt;Easy&lt;/td&gt;
&lt;td&gt;Hard&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Performance&lt;/td&gt;
&lt;td&gt;Can be slower&lt;/td&gt;
&lt;td&gt;Faster&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Best for&lt;/td&gt;
&lt;td&gt;Complex forms&lt;/td&gt;
&lt;td&gt;Simple forms&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;




&lt;h2&gt;
  
  
  Which One Should You Use?
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Use &lt;strong&gt;Controlled Forms&lt;/strong&gt; when:
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;You need real-time validation&lt;/li&gt;
&lt;li&gt;UI depends on input values&lt;/li&gt;
&lt;li&gt;You’re building dashboards, admin panels&lt;/li&gt;
&lt;li&gt;You’re using form libraries (Formik, React Hook Form)&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  Use &lt;strong&gt;Uncontrolled Forms&lt;/strong&gt; when:
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;Simple forms&lt;/li&gt;
&lt;li&gt;Performance-sensitive inputs&lt;/li&gt;
&lt;li&gt;One-time submission&lt;/li&gt;
&lt;li&gt;Legacy HTML forms&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Final Thoughts
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Start simple → uncontrolled&lt;/li&gt;
&lt;li&gt;Add complexity → controlled&lt;/li&gt;
&lt;li&gt;Don’t blindly control everything&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;React gives you &lt;strong&gt;choices&lt;/strong&gt;—good engineers know &lt;strong&gt;when&lt;/strong&gt; to use each.&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>typescript</category>
      <category>react</category>
    </item>
    <item>
      <title>Next.js</title>
      <dc:creator>Kiran Balaji</dc:creator>
      <pubDate>Sat, 08 Mar 2025 13:25:22 +0000</pubDate>
      <link>https://dev.to/kiran_balaji_197/nextjs-4924</link>
      <guid>https://dev.to/kiran_balaji_197/nextjs-4924</guid>
      <description></description>
      <category>nextjs</category>
    </item>
  </channel>
</rss>
