genui_catalog 0.2.1
genui_catalog: ^0.2.1 copied to clipboard
Ready-to-use GenUI CatalogItems for building dynamic AI-generated UIs.
import 'package:flutter/material.dart';
import 'package:genui_catalog_example/utils/constants.dart';
import 'package:url_launcher/url_launcher.dart';
import 'screens/home_screen.dart';
import 'screens/ai_demo_screen.dart';
import 'screens/data_screen.dart';
import 'screens/workflow_screen.dart';
import 'screens/forms_screen.dart';
import 'screens/media_screen.dart';
void main() {
runApp(const GenUICatalogApp());
}
class GenUICatalogApp extends StatelessWidget {
const GenUICatalogApp({super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'GenUI Catalog — Demo',
debugShowCheckedModeBanner: false,
theme: ThemeData(
useMaterial3: true,
colorScheme: ColorScheme.fromSeed(
seedColor: const Color(0xFF5C35CC),
brightness: Brightness.light,
),
fontFamily: 'Roboto',
),
home: const AppShell(),
);
}
}
// ---------------------------------------------------------------------------
// Navigation model
// ---------------------------------------------------------------------------
class _NavItem {
final String label;
final IconData icon;
final IconData activeIcon;
const _NavItem({
required this.label,
required this.icon,
required this.activeIcon,
});
}
const List<_NavItem> _navItems = [
_NavItem(label: 'Home', icon: Icons.home_outlined, activeIcon: Icons.home),
_NavItem(
label: 'AI Demo',
icon: Icons.auto_awesome_outlined,
activeIcon: Icons.auto_awesome,
),
_NavItem(
label: 'Data',
icon: Icons.bar_chart_outlined,
activeIcon: Icons.bar_chart,
),
_NavItem(
label: 'Workflow',
icon: Icons.account_tree_outlined,
activeIcon: Icons.account_tree,
),
_NavItem(
label: 'Forms',
icon: Icons.dynamic_form_outlined,
activeIcon: Icons.dynamic_form,
),
_NavItem(
label: 'Media',
icon: Icons.photo_library_outlined,
activeIcon: Icons.photo_library,
),
];
// ---------------------------------------------------------------------------
// App Shell — responsive layout
// ---------------------------------------------------------------------------
class AppShell extends StatefulWidget {
const AppShell({super.key});
@override
State<AppShell> createState() => AppShellState();
}
class AppShellState extends State<AppShell> {
int _selectedIndex = 0;
void navigateTo(int index) {
setState(() => _selectedIndex = index);
}
static const _screens = [
HomeScreen(),
AiDemoScreen(),
DataScreen(),
WorkflowScreen(),
FormsScreen(),
MediaScreen(),
];
@override
Widget build(BuildContext context) {
return LayoutBuilder(
builder: (context, constraints) {
final isDesktop = constraints.maxWidth >= 900;
if (isDesktop) {
return _DesktopLayout(
selectedIndex: _selectedIndex,
onNavTap: (i) => setState(() => _selectedIndex = i),
screens: _screens,
);
}
return _MobileLayout(
selectedIndex: _selectedIndex,
onNavTap: (i) => setState(() => _selectedIndex = i),
screens: _screens,
);
},
);
}
}
// ---------------------------------------------------------------------------
// Desktop layout
// ---------------------------------------------------------------------------
class _DesktopLayout extends StatelessWidget {
final int selectedIndex;
final ValueChanged<int> onNavTap;
final List<Widget> screens;
const _DesktopLayout({
required this.selectedIndex,
required this.onNavTap,
required this.screens,
});
@override
Widget build(BuildContext context) {
final colorScheme = Theme.of(context).colorScheme;
return Scaffold(
body: Row(
children: [
// Sidebar
SizedBox(
width: 240,
child: Container(
decoration: BoxDecoration(
color: colorScheme.surface,
border: Border(
right: BorderSide(
color: colorScheme.outlineVariant,
width: 1,
),
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Logo / brand
Padding(
padding: const EdgeInsets.fromLTRB(20, 28, 20, 12),
child: Row(
children: [
Container(
width: 36,
height: 36,
decoration: BoxDecoration(
gradient: LinearGradient(
colors: [
colorScheme.primary,
colorScheme.tertiary,
],
begin: Alignment.topLeft,
end: Alignment.bottomRight,
),
borderRadius: BorderRadius.circular(8),
),
child: const Icon(
Icons.grid_view_rounded,
color: Colors.white,
size: 20,
),
),
const SizedBox(width: 12),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
'GenUI',
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.w800,
color: colorScheme.primary,
letterSpacing: -0.3,
),
),
Text(
'Catalog',
style: TextStyle(
fontSize: 12,
fontWeight: FontWeight.w500,
color: colorScheme.onSurfaceVariant,
letterSpacing: 0.2,
),
),
],
),
],
),
),
Padding(
padding: const EdgeInsets.symmetric(horizontal: 16),
child: Divider(
height: 1,
color: colorScheme.outlineVariant,
),
),
const SizedBox(height: 8),
// Nav items
Expanded(
child: ListView.builder(
padding: const EdgeInsets.symmetric(
horizontal: 12,
vertical: 4,
),
itemCount: _navItems.length,
itemBuilder: (context, i) {
final item = _navItems[i];
final isSelected = i == selectedIndex;
return _SidebarNavItem(
item: item,
isSelected: isSelected,
onTap: () => onNavTap(i),
);
},
),
),
// Bottom links
Padding(
padding: const EdgeInsets.symmetric(horizontal: 12),
child: Divider(
height: 1,
color: colorScheme.outlineVariant,
),
),
Padding(
padding: const EdgeInsets.all(12),
child: Column(
children: [
_SidebarLinkButton(
icon: Icons.public,
label: 'pub.dev',
onTap: () => launchUrl(Uri.parse(pubUrl)),
),
const SizedBox(height: 4),
_SidebarLinkButton(
icon: Icons.code,
label: 'GitHub',
onTap: () => launchUrl(Uri.parse(githubUrl)),
),
],
),
),
],
),
),
),
// Content area
Expanded(
child: IndexedStack(index: selectedIndex, children: screens),
),
],
),
);
}
}
class _SidebarNavItem extends StatelessWidget {
final _NavItem item;
final bool isSelected;
final VoidCallback onTap;
const _SidebarNavItem({
required this.item,
required this.isSelected,
required this.onTap,
});
@override
Widget build(BuildContext context) {
final colorScheme = Theme.of(context).colorScheme;
return Padding(
padding: const EdgeInsets.symmetric(vertical: 2),
child: Material(
color: isSelected ? colorScheme.primaryContainer : Colors.transparent,
borderRadius: BorderRadius.circular(10),
child: InkWell(
onTap: onTap,
borderRadius: BorderRadius.circular(10),
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 10),
child: Row(
children: [
Icon(
isSelected ? item.activeIcon : item.icon,
size: 20,
color: isSelected
? colorScheme.primary
: colorScheme.onSurfaceVariant,
),
const SizedBox(width: 12),
Text(
item.label,
style: TextStyle(
fontSize: 14,
fontWeight: isSelected
? FontWeight.w600
: FontWeight.normal,
color: isSelected
? colorScheme.primary
: colorScheme.onSurface,
),
),
],
),
),
),
),
);
}
}
class _SidebarLinkButton extends StatelessWidget {
final IconData icon;
final String label;
final VoidCallback onTap;
const _SidebarLinkButton({
required this.icon,
required this.label,
required this.onTap,
});
@override
Widget build(BuildContext context) {
final colorScheme = Theme.of(context).colorScheme;
return Material(
color: Colors.transparent,
borderRadius: BorderRadius.circular(8),
child: InkWell(
onTap: onTap,
borderRadius: BorderRadius.circular(8),
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8),
child: Row(
children: [
Icon(icon, size: 16, color: colorScheme.onSurfaceVariant),
const SizedBox(width: 10),
Text(
label,
style: TextStyle(
fontSize: 13,
color: colorScheme.onSurfaceVariant,
),
),
const Spacer(),
Icon(
Icons.open_in_new,
size: 12,
color: colorScheme.onSurfaceVariant.withValues(alpha: 0.6),
),
],
),
),
),
);
}
}
// ---------------------------------------------------------------------------
// Mobile layout
// ---------------------------------------------------------------------------
class _MobileLayout extends StatelessWidget {
final int selectedIndex;
final ValueChanged<int> onNavTap;
final List<Widget> screens;
const _MobileLayout({
required this.selectedIndex,
required this.onNavTap,
required this.screens,
});
@override
Widget build(BuildContext context) {
final colorScheme = Theme.of(context).colorScheme;
return Scaffold(
appBar: AppBar(
title: Row(
mainAxisSize: MainAxisSize.min,
children: [
Container(
width: 28,
height: 28,
decoration: BoxDecoration(
gradient: LinearGradient(
colors: [colorScheme.primary, colorScheme.tertiary],
begin: Alignment.topLeft,
end: Alignment.bottomRight,
),
borderRadius: BorderRadius.circular(6),
),
child: const Icon(
Icons.grid_view_rounded,
color: Colors.white,
size: 16,
),
),
const SizedBox(width: 10),
Text(
'GenUI Catalog',
style: TextStyle(
fontWeight: FontWeight.w700,
color: colorScheme.primary,
fontSize: 18,
),
),
],
),
backgroundColor: colorScheme.surface,
surfaceTintColor: colorScheme.primary,
bottom: PreferredSize(
preferredSize: const Size.fromHeight(1),
child: Divider(height: 1, color: colorScheme.outlineVariant),
),
),
body: IndexedStack(index: selectedIndex, children: screens),
bottomNavigationBar: NavigationBar(
selectedIndex: selectedIndex,
onDestinationSelected: onNavTap,
destinations: _navItems
.map(
(item) => NavigationDestination(
icon: Icon(item.icon),
selectedIcon: Icon(item.activeIcon),
label: item.label,
),
)
.toList(),
),
);
}
}