mobile-responsiveness
Scannednpx machina-cli add skill Makiya1202/ai-agents-skills/mobile-responsiveness --openclawFiles (1)
SKILL.md
5.2 KB
Mobile Responsiveness
Build responsive, mobile-first web applications.
Mobile-First Breakpoints
/* Mobile first - no media query needed for mobile base */
.container {
padding: 1rem;
}
/* Tablet */
@media (min-width: 768px) {
.container {
padding: 2rem;
}
}
/* Desktop */
@media (min-width: 1024px) {
.container {
padding: 3rem;
max-width: 1200px;
margin: 0 auto;
}
}
/* Large desktop */
@media (min-width: 1280px) {
.container {
max-width: 1400px;
}
}
Tailwind Breakpoints
<div className="
p-4 /* Mobile: padding 1rem */
md:p-8 /* Tablet 768px+: padding 2rem */
lg:p-12 /* Desktop 1024px+: padding 3rem */
xl:max-w-6xl /* Large 1280px+: max-width */
">
<h1 className="
text-2xl /* Mobile */
md:text-3xl /* Tablet */
lg:text-4xl /* Desktop */
">
Responsive Heading
</h1>
</div>
Fluid Typography
:root {
/* Fluid font size: 16px at 320px viewport, 20px at 1200px viewport */
--font-size-base: clamp(1rem, 0.9rem + 0.5vw, 1.25rem);
/* Fluid heading */
--font-size-h1: clamp(2rem, 1.5rem + 2.5vw, 4rem);
}
body {
font-size: var(--font-size-base);
}
h1 {
font-size: var(--font-size-h1);
}
Touch Interactions
import { useState } from 'react';
function SwipeableCard({ onSwipeLeft, onSwipeRight, children }) {
const [touchStart, setTouchStart] = useState<number | null>(null);
const [touchEnd, setTouchEnd] = useState<number | null>(null);
const minSwipeDistance = 50;
const onTouchStart = (e: React.TouchEvent) => {
setTouchEnd(null);
setTouchStart(e.targetTouches[0].clientX);
};
const onTouchMove = (e: React.TouchEvent) => {
setTouchEnd(e.targetTouches[0].clientX);
};
const onTouchEnd = () => {
if (!touchStart || !touchEnd) return;
const distance = touchStart - touchEnd;
const isLeftSwipe = distance > minSwipeDistance;
const isRightSwipe = distance < -minSwipeDistance;
if (isLeftSwipe) onSwipeLeft?.();
if (isRightSwipe) onSwipeRight?.();
};
return (
<div
onTouchStart={onTouchStart}
onTouchMove={onTouchMove}
onTouchEnd={onTouchEnd}
>
{children}
</div>
);
}
Mobile Navigation
import { useState } from 'react';
function MobileNav() {
const [isOpen, setIsOpen] = useState(false);
return (
<>
{/* Hamburger button - visible on mobile */}
<button
className="md:hidden p-2"
onClick={() => setIsOpen(!isOpen)}
aria-expanded={isOpen}
aria-label="Toggle menu"
>
<span className={`hamburger ${isOpen ? 'open' : ''}`} />
</button>
{/* Mobile menu */}
<nav
className={`
fixed inset-0 bg-white z-50 transform transition-transform
${isOpen ? 'translate-x-0' : '-translate-x-full'}
md:static md:translate-x-0 md:bg-transparent
`}
>
<ul className="flex flex-col md:flex-row gap-4 p-4 md:p-0">
<li><a href="/">Home</a></li>
<li><a href="/about">About</a></li>
<li><a href="/contact">Contact</a></li>
</ul>
</nav>
{/* Backdrop */}
{isOpen && (
<div
className="fixed inset-0 bg-black/50 z-40 md:hidden"
onClick={() => setIsOpen(false)}
/>
)}
</>
);
}
Safe Areas (Notch/Home Indicator)
/* Account for iPhone notch and home indicator */
.container {
padding-left: env(safe-area-inset-left);
padding-right: env(safe-area-inset-right);
padding-bottom: env(safe-area-inset-bottom);
}
/* Fixed bottom navigation */
.bottom-nav {
position: fixed;
bottom: 0;
left: 0;
right: 0;
padding-bottom: max(1rem, env(safe-area-inset-bottom));
}
Viewport Meta Tag
<meta
name="viewport"
content="width=device-width, initial-scale=1, viewport-fit=cover"
/>
useMediaQuery Hook
import { useState, useEffect } from 'react';
function useMediaQuery(query: string): boolean {
const [matches, setMatches] = useState(false);
useEffect(() => {
const media = window.matchMedia(query);
setMatches(media.matches);
const listener = (e: MediaQueryListEvent) => setMatches(e.matches);
media.addEventListener('change', listener);
return () => media.removeEventListener('change', listener);
}, [query]);
return matches;
}
// Usage
function Component() {
const isMobile = useMediaQuery('(max-width: 767px)');
const isTablet = useMediaQuery('(min-width: 768px) and (max-width: 1023px)');
const isDesktop = useMediaQuery('(min-width: 1024px)');
return isMobile ? <MobileView /> : <DesktopView />;
}
Resources
- Responsive Design: https://web.dev/learn/design/
- Mobile-First CSS: https://developer.mozilla.org/en-US/docs/Learn/CSS/CSS_layout/Responsive_Design
- Viewport Units: https://developer.mozilla.org/en-US/docs/Web/CSS/length#viewport-percentage_lengths
Source
git clone https://github.com/Makiya1202/ai-agents-skills/blob/master/skills/mobile-responsiveness/SKILL.mdView on GitHub Overview
Build responsive, mobile-first web apps that adapt to phones, tablets, and desktops. This skill covers mobile breakpoints with CSS, Tailwind utilities, fluid typography, touch interactions, mobile navigation patterns, and safe-area considerations.
How This Skill Works
Start with a mobile base stylesheet and progressively enhance with breakpoints via media queries or Tailwind responsive classes. Use CSS clamp and custom properties to achieve fluid typography. Implement touch interactions with components like a SwipeableCard, a mobile navigation pattern, and safe-area env utilities for notches.
When to Use It
- Design responsive layouts that adapt to phones, tablets, and desktops using mobile-first breakpoints.
- Add touch interactions such as swipe gestures for cards or panels.
- Implement a mobile navigation pattern with a hamburger menu and off-canvas drawer.
- Use fluid typography to scale text smoothly across viewport sizes.
- Respect safe-area insets for notches and home indicators on mobile devices.
Quick Start
- Step 1: Build mobile base styles with mobile padding and typography values
- Step 2: Add breakpoints with media queries or Tailwind utilities for tablet and desktop
- Step 3: Implement touch interactions, mobile navigation, and safe-area handling; test across devices
Best Practices
- Start with a mobile layout and progressively add breakpoints for larger viewports.
- Use CSS clamp and root variables to create fluid typography with scalable sizes.
- Leverage Tailwind responsive utilities or CSS media queries for clean breakpoints.
- Build accessible touch interactions with clear thresholds and keyboard support when possible.
- Account for safe area insets using env(safe-area-inset-*) in CSS.
Example Use Cases
- SwipeableCard component that triggers actions on left or right swipes
- MobileNav pattern with a hamburger button and sliding drawer
- CSS mobile base with tablet and desktop media queries for padding and max-width
- Tailwind example using md, lg, xl classes to adjust padding and heading sizes
- Safe Areas snippet applying env(safe-area-inset-*) to container padding
Frequently Asked Questions
Add this skill to your agents