hebrew-rtl-best-practices
Original:🇺🇸 English
Translated
Implement right-to-left (RTL) layouts for Hebrew web and mobile applications. Use when user asks about RTL layout, Hebrew text direction, bidirectional (bidi) text, Hebrew CSS, "right to left", or needs to build Hebrew UI. Covers CSS logical properties, Tailwind RTL, React/Next.js RTL setup, Hebrew typography, and font selection. Do NOT use for Arabic RTL (similar but different typography) unless user explicitly asks for shared RTL patterns.
4installs
Sourceskills-il/localization
Added on
NPX Install
npx skill4agent add skills-il/localization hebrew-rtl-best-practicesTags
Translated version includes tags in frontmatterSKILL.md Content
View Translation Comparison →Hebrew RTL Best Practices
Instructions
Step 1: Set Up Document Direction
Always start with the HTML attribute (not just CSS):
html
<html lang="he" dir="rtl">This tells browsers, screen readers, and CSS to use RTL as the base direction.
Step 2: Use CSS Logical Properties
NEVER use physical directional properties for layout:
| Physical (avoid) | Logical (use) |
|---|---|
| |
| |
| |
| |
| |
| |
| |
| |
| |
This ensures the layout automatically mirrors in RTL mode.
Step 3: Handle Bidirectional Text
When mixing Hebrew and English/numbers:
css
/* Isolate embedded LTR content */
.ltr-content {
unicode-bidi: isolate;
direction: ltr;
}
/* For inline elements with mixed content */
.bidi-override {
unicode-bidi: bidi-override;
}Common bidi issues:
- Phone numbers appearing reversed: Wrap in
<bdo dir="ltr"> - Punctuation at wrong end of sentence: Use
unicode-bidi: isolate - URLs/emails in Hebrew text: Wrap in
<span dir="ltr">
Step 4: Hebrew Typography
Recommended font stack:
css
font-family: 'Heebo', 'Assistant', 'Rubik', 'Noto Sans Hebrew', sans-serif;Typography settings:
css
body[dir="rtl"] {
font-size: 16px; /* Hebrew needs slightly larger than Latin */
line-height: 1.7;
letter-spacing: normal; /* NEVER add letter-spacing for Hebrew */
word-spacing: 0.05em; /* Slight word spacing improves readability */
}Step 5: Framework-Specific Setup
Tailwind CSS RTL (v3.3+ / v4):
Prefer logical property utilities over / variants:
rtl:ltr:| Physical class | Logical class | CSS property |
|---|---|---|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
html
<!-- Bad: requires two classes, breaks without dir attribute -->
<div class="ltr:ml-4 rtl:mr-4">...</div>
<!-- Good: single class, auto-mirrors based on dir -->
<div class="ms-4">...</div>Reserve / variants only for cases logical properties cannot handle (e.g., directional icons, transforms).
rtl:ltr:Tailwind v4 note: v4 uses CSS-first configuration ( in CSS) instead of . Logical utilities work identically in both v3 and v4.
@import "tailwindcss"tailwind.config.jsNext.js App Router:
tsx
// app/layout.tsx
import { Heebo } from 'next/font/google';
const heebo = Heebo({
subsets: ['hebrew', 'latin'],
weight: ['400', '500', '700'],
});
export default async function RootLayout({
children,
params,
}: {
children: React.ReactNode;
params: Promise<{ locale: string }>;
}) {
const { locale } = await params;
const isRTL = locale === 'he';
return (
<html lang={locale} dir={isRTL ? 'rtl' : 'ltr'}>
<body className={heebo.className}>{children}</body>
</html>
);
}next/fontReact with MUI:
jsx
import { createTheme, ThemeProvider } from '@mui/material/styles';
import { CacheProvider } from '@emotion/react';
import createCache from '@emotion/cache';
import rtlPlugin from 'stylis-plugin-rtl';
import { prefixer } from 'stylis';
const cacheRtl = createCache({
key: 'muirtl',
stylisPlugins: [prefixer, rtlPlugin],
});
const theme = createTheme({ direction: 'rtl' });Step 6: Common Pitfalls to Check
- Icons with directional meaning (arrows, back buttons) -- mirror them
- Progress bars -- should fill from right to left
- Sliders/carousels -- swipe direction should reverse
- Form labels -- should be right-aligned
- Breadcrumbs -- separator direction should reverse
- Tables -- header alignment and cell alignment
- Charts -- x-axis may need to reverse for Hebrew readers
Examples
Example 1: Convert LTR Component to RTL
User says: "Make this card component work in Hebrew"
Before (LTR-only):
css
.card {
margin-left: 16px;
padding-right: 12px;
text-align: left;
border-left: 3px solid blue;
}After (RTL-compatible):
css
.card {
margin-inline-start: 16px;
padding-inline-end: 12px;
text-align: start;
border-inline-start: 3px solid blue;
}With Tailwind, replace with .
ml-4 pr-3 text-left border-l-4ms-4 pe-3 text-start border-s-4Example 2: Bidi Text Issue
User says: "Numbers are showing backwards in my Hebrew text"
html
<!-- Wrong: phone number renders as 0544-123-050 -->
<p>התקשרו אלינו: 050-321-4450</p>
<!-- Correct: isolate the LTR content -->
<p>התקשרו אלינו: <span dir="ltr">050-321-4450</span></p>Use on the containing span for CSS-only solutions.
unicode-bidi: isolateExample 3: Tailwind RTL Navigation
User says: "My sidebar is on the wrong side in Hebrew"
html
<!-- Bad: sidebar stuck on left -->
<aside class="fixed left-0 w-64">...</aside>
<!-- Good: sidebar auto-mirrors -->
<aside class="fixed start-0 w-64">...</aside>
<!-- Back arrow icon still needs rtl: variant -->
<button class="rtl:rotate-180">
<ArrowLeftIcon />
</button>Bundled Resources
References
- — Complete physical-to-logical CSS property mapping table (margin, padding, border, positioning, text alignment, sizing) plus Hebrew font stack recommendations for sans-serif, serif, and monospace. Consult when converting any LTR stylesheet to RTL-compatible logical properties or choosing Hebrew web fonts.
references/css-logical-properties.md
Gotchas
- CSS is wrong for Hebrew. Use
text-align: leftwhich respects the document direction. Agents frequently hardcodetext-align: startalignment in CSS.left - and
margin-leftdo not flip in RTL mode. Use CSS logical properties:padding-rightandmargin-inline-startinstead. Agents trained on LTR CSS will generate physical properties.padding-inline-end - Flexbox direction auto-reverses in RTL, but
rowalso reverses, causing a double-flip back to LTR order. Agents may addrow-reversethinking it creates RTL, but it actually creates LTR within an RTL context.row-reverse - Phone numbers, credit card numbers, and code snippets must remain LTR even inside RTL containers. Wrap them in or use
<bdo dir="ltr">on the containing element. Agents often let these inherit RTL.direction: ltr
Reference Links
| Source | URL | What to Check |
|---|---|---|
| MDN CSS Logical Properties | https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_logical_properties_and_values | Full property list, browser support tables |
| Tailwind CSS RTL Support | https://tailwindcss.com/docs/hover-focus-and-other-states#rtl-support | |
| Tailwind Logical Properties | https://tailwindcss.com/docs/margin#logical-properties | |
| Google Fonts Hebrew | https://fonts.google.com/?subset=hebrew | Available Hebrew font families |
| W3C Internationalization | https://www.w3.org/International/articles/inline-bidi-markup/ | Unicode bidi algorithm, markup best practices |
Troubleshooting
Error: "Text alignment looks wrong"
Cause: Using instead of
Solution: Replace all / in text-align with /.
text-align: lefttext-align: startleftrightstartendError: "Layout not mirroring"
Cause: Using physical margin/padding instead of logical properties
Solution: Replace all / with /.
margin-leftmargin-rightmargin-inline-startmargin-inline-end