https://github.com/skiniks/react-native-shiki-engine
Shiki syntax highlighting for React Native. Zero bridge overhead with native Oniguruma regex engine.
https://github.com/skiniks/react-native-shiki-engine
android cpp ios javascript jsi native react react-native shiki syntax-highlighting textmate turbo-modules typescript
Last synced: 6 months ago
JSON representation
Shiki syntax highlighting for React Native. Zero bridge overhead with native Oniguruma regex engine.
- Host: GitHub
- URL: https://github.com/skiniks/react-native-shiki-engine
- Owner: skiniks
- License: mit
- Created: 2024-11-25T19:14:10.000Z (11 months ago)
- Default Branch: main
- Last Pushed: 2025-03-23T12:59:44.000Z (7 months ago)
- Last Synced: 2025-03-23T13:38:46.836Z (7 months ago)
- Topics: android, cpp, ios, javascript, jsi, native, react, react-native, shiki, syntax-highlighting, textmate, turbo-modules, typescript
- Language: C
- Homepage:
- Size: 6.23 MB
- Stars: 53
- Watchers: 1
- Forks: 0
- Open Issues: 9
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- Funding: .github/FUNDING.yml
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# React Native Shiki Engine
Oniguruma regex engine implementation for React Native, providing high-performance syntax highlighting capabilities through [Shiki](https://github.com/shikijs/shiki). This module implements a JSI-based native bridge to Oniguruma, enabling efficient pattern matching and syntax highlighting in React Native applications.
> [!IMPORTANT]
> React Native Shiki Engine requires the New Architecture to be enabled (react-native 0.71+)## Features
- **High-performance** native regex engine using Oniguruma, optimized for syntax highlighting
- **Fully synchronous** pattern matching with no async/await, no Promises, no Bridge
- Uses [**JSI**](https://reactnative.dev/docs/the-new-architecture/landing-page#fast-javascriptnative-interfacing) and [**C++ TurboModules**](https://github.com/reactwg/react-native-new-architecture/blob/main/docs/turbo-modules-xplat.md) for direct JavaScript-to-native communication
- **Smart pattern caching** system for optimal performance
- **Memory efficient** with automatic cleanup of unused patterns
- **Full compatibility** with Shiki's regex engine requirements
- Written in **modern C++** with robust memory management
- **Multiple pattern** support with efficient cache management## Installation
### React Native
```sh
yarn add react-native-shiki-engine
cd ios && pod install
```### Expo
```sh
npx expo install react-native-shiki-engine
npx expo prebuild
```## Usage
### Basic Integration
```tsx
import { createHighlighterCore } from '@shikijs/core'
import { createNativeEngine, isNativeEngineAvailable } from 'react-native-shiki-engine'
import javascript from 'shiki/dist/langs/javascript.mjs'
import nord from 'shiki/dist/themes/nord.mjs'if (!isNativeEngineAvailable()) {
throw new Error('Native engine is not available')
}const highlighter = createHighlighterCore({
themes: ['nord'],
langs: ['javascript'],
engine: createNativeEngine(),
})
```> [!IMPORTANT]
> ### Performance Note: The Highlighter Instance
>
> Create and maintain a single Highlighter instance at the app level. Avoid instantiating new highlighters inside components or frequently called functions.
>
> #### Recommended:
> - Store the highlighter instance in a global singleton or context
> - Initialize it during app startup
> - Reuse the same instance across your entire React Native application
>
> #### Not Recommended:
> - Creating new instances inside component render methods
> - Initializing highlighters inside useEffect or event handlers
> - Multiple instances for the same language/theme combination
>
> See the [example directory](https://github.com/skiniks/react-native-shiki-engine/tree/main/example) for a reference implementation using React Context to maintain a single highlighter instance.### Advanced Configuration
The native engine supports configuration options to optimize performance:
```typescript
createNativeEngine({
// Maximum number of patterns to cache
maxCacheSize: 1000,
})
```## Web Platform Support
For web applications, this native engine is not compatible as it relies on React Native's TurboModules and JSI. To support web platforms, you can use Shiki's default WASM or JavaScript engine instead.
1. Use Shiki's default WASM/JavaScript engine for web platforms
2. Implement platform-specific engine selection in your code### Example Implementation
```typescript
import { createHighlighterCore } from '@shikijs/core'
import { createOnigurumaEngine } from '@shikijs/engine-oniguruma'
import { createNativeEngine, isNativeEngineAvailable } from 'react-native-shiki-engine'function createAdaptiveEngine() {
// Use native engine for native platforms
if (isNativeEngineAvailable()) {
return createNativeEngine()
}
// Fallback to Shiki's WASM engine for web platforms
return createOnigurumaEngine(import('shiki/wasm'))
}const highlighter = createHighlighterCore({
themes: ['nord'],
langs: ['javascript'],
engine: createAdaptiveEngine(),
})
```This approach ensures:
- Optimal performance on React Native using the native engine
- Seamless fallback to Shiki's WASM engine on web platforms
- Cross-platform compatibility with a single codebase## Technical Architecture
### Native Implementation
The module uses a three-layer architecture optimizing for both performance and developer experience:
1. **JavaScript Layer** (`src/`)
- TypeScript interfaces and JS wrapper for type safety
- Efficient pattern lifecycle management with automatic cleanup
- Seamless integration with Shiki's API
- Error boundaries with graceful degradation2. **JSI Bridge** (`cpp/`)
- Zero-copy JavaScript-to-native communication
- Smart pointer-based memory management
- Thread-safe pattern caching with LRU eviction
- Host object lifetime tracking3. **Oniguruma Core** (vendored)
- High-performance native regex engine
- Optimized pattern matching with capture groups
- Full Unicode support with UTF-8/16 encoding
- Non-backtracking algorithm for predictable performance### Pattern Caching
The engine implements a sophisticated multi-level caching system:
- **L1 Cache**: Hot patterns in JSI host objects
- Zero-copy access from JavaScript
- Reference-counted lifetime management
- Automatic cleanup on context destruction- **L2 Cache**: Compiled patterns in native memory
- LRU eviction with generational collection
- Adaptive sizing based on memory pressure
- Thread-safe concurrent access
- Configurable eviction policies- **Memory Management**
- Proactive cleanup of unused patterns
- Automatic defragmentation
- Memory pressure handling
- Configurable high/low watermarks## Supported Platforms
| Platform | Architecture | Description | Status |
| :-----------: | :----------: | :----------------------------------: | :----: |
| iOS Simulator | x86_64 | Intel-based simulators | ✅ |
| iOS Simulator | arm64 | Apple Silicon simulators | ✅ |
| iOS Device | arm64 | All modern iOS devices | ✅ |
| Android | arm64-v8a | Modern Android devices (64-bit ARM) | ✅ |
| Android | armeabi-v7a | Older Android devices (32-bit ARM) | ✅ |
| Android | x86 | Android emulators (32-bit Intel/AMD) | ✅ |
| Android | x86_64 | Android emulators (64-bit Intel/AMD) | ✅ |## Contributing
Contributions are welcome! Please read our [Contributing Guide](CONTRIBUTING.md) for details on:
- Code of conduct
- Development workflow
- Pull request process
- Coding standards## License
MIT License - see the [LICENSE](LICENSE) file for details
## Acknowledgments
- [Shiki](https://github.com/shikijs/shiki) - The beautiful yet powerful syntax highlighter we bring to React Native
- [Oniguruma](https://github.com/kkos/oniguruma) - The blazing-fast regex engine powering our native implementation
- [React Native](https://reactnative.dev/) - Making native module development possible with JSI and TurboModules## Support
For questions, bug reports, or feature requests:
- [GitHub Issues](https://github.com/skiniks/react-native-shiki-engine/issues)
- [GitHub Discussions](https://github.com/skiniks/react-native-shiki-engine/discussions)