openuispec 0.2.10 → 0.2.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (237) hide show
  1. package/README.md +3 -1
  2. package/check/index.ts +17 -0
  3. package/cli/index.ts +21 -3
  4. package/cli/init.ts +224 -10
  5. package/docs/cli.md +13 -8
  6. package/docs/file-formats.md +36 -0
  7. package/docs/implementation-notes.md +7 -0
  8. package/drift/index.ts +281 -40
  9. package/mcp-server/index.ts +179 -119
  10. package/mcp-server/screenshot.ts +19 -4
  11. package/package.json +5 -2
  12. package/prepare/index.ts +155 -18
  13. package/schema/openuispec.schema.json +59 -0
  14. package/schema/semantic-lint.ts +25 -1
  15. package/scripts/take-all-screenshots.ts +507 -0
  16. package/spec/openuispec-v0.1.md +13 -0
  17. package/status/index.ts +72 -2
  18. package/examples/social-app/.mcp.json +0 -10
  19. package/examples/social-app/AGENTS.md +0 -124
  20. package/examples/social-app/CLAUDE.md +0 -124
  21. package/examples/social-app/backend/.gitkeep +0 -1
  22. package/examples/social-app/generated/android/social-app/app/.paparazzi-hashes.json +0 -3
  23. package/examples/social-app/generated/android/social-app/app/build.gradle.kts +0 -94
  24. package/examples/social-app/generated/android/social-app/app/src/main/AndroidManifest.xml +0 -26
  25. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/AppContainer.kt +0 -20
  26. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/MainActivity.kt +0 -35
  27. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/SocialAppApplication.kt +0 -13
  28. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/MockData.kt +0 -98
  29. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/AppPreferences.kt +0 -19
  30. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/DataStorePreferencesRepository.kt +0 -68
  31. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/PreferencesRepository.kt +0 -15
  32. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/model/Models.kt +0 -34
  33. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/MainShell.kt +0 -390
  34. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/Components.kt +0 -234
  35. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/ContractPrimitives.kt +0 -641
  36. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/navigation/RootComponent.kt +0 -113
  37. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ChatDetailScreen.kt +0 -212
  38. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/CreatePostScreen.kt +0 -113
  39. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/DiscoverScreen.kt +0 -137
  40. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/EditProfileScreen.kt +0 -180
  41. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/HomeFeedScreen.kt +0 -169
  42. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/MessagesInboxScreen.kt +0 -85
  43. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/NotificationsScreen.kt +0 -74
  44. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/PostDetailScreen.kt +0 -293
  45. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ProfileSelfScreen.kt +0 -116
  46. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SearchResultsScreen.kt +0 -161
  47. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsScreen.kt +0 -164
  48. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsStore.kt +0 -95
  49. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/UserProfileScreen.kt +0 -123
  50. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Color.kt +0 -33
  51. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Shape.kt +0 -41
  52. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Spacing.kt +0 -20
  53. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Theme.kt +0 -82
  54. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Type.kt +0 -60
  55. package/examples/social-app/generated/android/social-app/app/src/main/res/drawable/ic_launcher_foreground.xml +0 -9
  56. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +0 -5
  57. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml +0 -5
  58. package/examples/social-app/generated/android/social-app/app/src/main/res/values/strings.xml +0 -91
  59. package/examples/social-app/generated/android/social-app/app/src/main/res/values/themes.xml +0 -10
  60. package/examples/social-app/generated/android/social-app/app/src/main/res/values-ru/strings.xml +0 -79
  61. package/examples/social-app/generated/android/social-app/app/src/main/res/values-uz/strings.xml +0 -79
  62. package/examples/social-app/generated/android/social-app/app/src/main/xml/AndroidManifest.xml +0 -23
  63. package/examples/social-app/generated/android/social-app/app/src/test/kotlin/com/social/app/screenshots/HomeFeedScreenshotTest.kt +0 -34
  64. package/examples/social-app/generated/android/social-app/build.gradle.kts +0 -7
  65. package/examples/social-app/generated/android/social-app/gradle/libs.versions.toml +0 -50
  66. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.jar +0 -0
  67. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.properties +0 -8
  68. package/examples/social-app/generated/android/social-app/gradle.properties +0 -11
  69. package/examples/social-app/generated/android/social-app/gradlew +0 -248
  70. package/examples/social-app/generated/android/social-app/settings.gradle.kts +0 -27
  71. package/examples/social-app/generated/web/social-app/index.html +0 -12
  72. package/examples/social-app/generated/web/social-app/package-lock.json +0 -2517
  73. package/examples/social-app/generated/web/social-app/package.json +0 -27
  74. package/examples/social-app/generated/web/social-app/src/app/App.tsx +0 -58
  75. package/examples/social-app/generated/web/social-app/src/components/Shell.tsx +0 -259
  76. package/examples/social-app/generated/web/social-app/src/components/cards.tsx +0 -317
  77. package/examples/social-app/generated/web/social-app/src/components/ui.tsx +0 -340
  78. package/examples/social-app/generated/web/social-app/src/flows/CreatePostFlow.tsx +0 -86
  79. package/examples/social-app/generated/web/social-app/src/i18n.tsx +0 -59
  80. package/examples/social-app/generated/web/social-app/src/lib/icons.tsx +0 -85
  81. package/examples/social-app/generated/web/social-app/src/lib/tokens.ts +0 -70
  82. package/examples/social-app/generated/web/social-app/src/lib/utils.ts +0 -97
  83. package/examples/social-app/generated/web/social-app/src/locales/en.json +0 -67
  84. package/examples/social-app/generated/web/social-app/src/locales/ru.json +0 -67
  85. package/examples/social-app/generated/web/social-app/src/locales/uz.json +0 -67
  86. package/examples/social-app/generated/web/social-app/src/main.tsx +0 -16
  87. package/examples/social-app/generated/web/social-app/src/screens/ChatDetailScreen.tsx +0 -90
  88. package/examples/social-app/generated/web/social-app/src/screens/DiscoverScreen.tsx +0 -86
  89. package/examples/social-app/generated/web/social-app/src/screens/EditProfileScreen.tsx +0 -57
  90. package/examples/social-app/generated/web/social-app/src/screens/HomeFeedScreen.tsx +0 -103
  91. package/examples/social-app/generated/web/social-app/src/screens/MessagesInboxScreen.tsx +0 -52
  92. package/examples/social-app/generated/web/social-app/src/screens/NotificationsScreen.tsx +0 -41
  93. package/examples/social-app/generated/web/social-app/src/screens/PostDetailScreen.tsx +0 -115
  94. package/examples/social-app/generated/web/social-app/src/screens/ProfileSelfScreen.tsx +0 -57
  95. package/examples/social-app/generated/web/social-app/src/screens/ProfileUserScreen.tsx +0 -76
  96. package/examples/social-app/generated/web/social-app/src/screens/SearchResultsScreen.tsx +0 -96
  97. package/examples/social-app/generated/web/social-app/src/screens/SettingsScreen.tsx +0 -79
  98. package/examples/social-app/generated/web/social-app/src/state/store.ts +0 -592
  99. package/examples/social-app/generated/web/social-app/src/styles.css +0 -125
  100. package/examples/social-app/generated/web/social-app/src/vite-env.d.ts +0 -1
  101. package/examples/social-app/generated/web/social-app/tsconfig.json +0 -22
  102. package/examples/social-app/generated/web/social-app/tsconfig.node.json +0 -13
  103. package/examples/social-app/generated/web/social-app/tsconfig.node.tsbuildinfo +0 -1
  104. package/examples/social-app/generated/web/social-app/tsconfig.tsbuildinfo +0 -1
  105. package/examples/social-app/generated/web/social-app/vite.config.d.ts +0 -2
  106. package/examples/social-app/generated/web/social-app/vite.config.js +0 -6
  107. package/examples/social-app/generated/web/social-app/vite.config.ts +0 -7
  108. package/examples/social-app/package.json +0 -13
  109. package/examples/social-app/take-web-screenshots.ts +0 -97
  110. package/examples/taskflow/.codex/config.toml +0 -4
  111. package/examples/taskflow/.mcp.json +0 -10
  112. package/examples/taskflow/AGENTS.md +0 -124
  113. package/examples/taskflow/CLAUDE.md +0 -124
  114. package/examples/taskflow/backend/.gitkeep +0 -1
  115. package/examples/taskflow/generated/android/TaskFlow/README.md +0 -43
  116. package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +0 -76
  117. package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +0 -1
  118. package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +0 -21
  119. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +0 -19
  120. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +0 -283
  121. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +0 -106
  122. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +0 -57
  123. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +0 -109
  124. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +0 -112
  125. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +0 -61
  126. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +0 -82
  127. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +0 -111
  128. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +0 -77
  129. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +0 -30
  130. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +0 -86
  131. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +0 -57
  132. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +0 -155
  133. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +0 -4
  134. package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +0 -5
  135. package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +0 -12
  136. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
  137. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +0 -7
  138. package/examples/taskflow/generated/android/TaskFlow/gradle.properties +0 -4
  139. package/examples/taskflow/generated/android/TaskFlow/gradlew +0 -18
  140. package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +0 -12
  141. package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +0 -18
  142. package/examples/taskflow/generated/ios/TaskFlow/README.md +0 -21
  143. package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +0 -115
  144. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +0 -24
  145. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +0 -150
  146. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +0 -220
  147. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +0 -122
  148. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +0 -21
  149. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +0 -201
  150. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +0 -48
  151. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +0 -59
  152. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +0 -63
  153. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +0 -85
  154. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +0 -219
  155. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +0 -320
  156. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +0 -41
  157. package/examples/taskflow/generated/ios/TaskFlow/project.yml +0 -31
  158. package/examples/taskflow/generated/web/TaskFlow/README.md +0 -19
  159. package/examples/taskflow/generated/web/TaskFlow/index.html +0 -12
  160. package/examples/taskflow/generated/web/TaskFlow/package-lock.json +0 -1908
  161. package/examples/taskflow/generated/web/TaskFlow/package.json +0 -24
  162. package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +0 -58
  163. package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +0 -55
  164. package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +0 -82
  165. package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +0 -191
  166. package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +0 -41
  167. package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +0 -131
  168. package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +0 -25
  169. package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +0 -39
  170. package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +0 -111
  171. package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +0 -13
  172. package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +0 -111
  173. package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +0 -82
  174. package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +0 -132
  175. package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +0 -105
  176. package/examples/taskflow/generated/web/TaskFlow/src/store.ts +0 -216
  177. package/examples/taskflow/generated/web/TaskFlow/src/styles.css +0 -617
  178. package/examples/taskflow/generated/web/TaskFlow/src/types.ts +0 -64
  179. package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +0 -78
  180. package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +0 -21
  181. package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +0 -6
  182. package/examples/todo-orbit/.codex/config.toml +0 -4
  183. package/examples/todo-orbit/.mcp.json +0 -10
  184. package/examples/todo-orbit/AGENTS.md +0 -124
  185. package/examples/todo-orbit/CLAUDE.md +0 -124
  186. package/examples/todo-orbit/backend/.gitkeep +0 -1
  187. package/examples/todo-orbit/generated/android/Todo Orbit/README.md +0 -14
  188. package/examples/todo-orbit/generated/android/Todo Orbit/app/build.gradle.kts +0 -58
  189. package/examples/todo-orbit/generated/android/Todo Orbit/app/proguard-rules.pro +0 -1
  190. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/AndroidManifest.xml +0 -20
  191. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/MainActivity.kt +0 -14
  192. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/TodoOrbitApp.kt +0 -345
  193. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/AppLogic.kt +0 -231
  194. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Models.kt +0 -169
  195. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Strings.kt +0 -8
  196. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/components/CommonComponents.kt +0 -236
  197. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/AnalyticsScreen.kt +0 -193
  198. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/SettingsScreen.kt +0 -102
  199. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/TasksScreen.kt +0 -347
  200. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/sheets/EditorSheets.kt +0 -347
  201. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/theme/TodoOrbitTheme.kt +0 -59
  202. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values/strings.xml +0 -149
  203. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values-ru/strings.xml +0 -155
  204. package/examples/todo-orbit/generated/android/Todo Orbit/build.gradle.kts +0 -4
  205. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.jar +0 -0
  206. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.properties +0 -7
  207. package/examples/todo-orbit/generated/android/Todo Orbit/gradle.properties +0 -4
  208. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew +0 -248
  209. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew.bat +0 -93
  210. package/examples/todo-orbit/generated/android/Todo Orbit/settings.gradle.kts +0 -18
  211. package/examples/todo-orbit/generated/ios/Todo Orbit/.screenshot-uitest/Sources/ScreenshotUITest.swift +0 -36
  212. package/examples/todo-orbit/generated/ios/Todo Orbit/README.md +0 -29
  213. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/en.lproj/Localizable.strings +0 -119
  214. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/ru.lproj/Localizable.strings +0 -119
  215. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/App/TodoOrbitApp.swift +0 -50
  216. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/OrbitChrome.swift +0 -204
  217. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/SchedulePreviewView.swift +0 -126
  218. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/TrendChartView.swift +0 -70
  219. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/RecurringRuleSheet.swift +0 -126
  220. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/TaskEditorSheet.swift +0 -61
  221. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Models/DomainModels.swift +0 -238
  222. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/AnalyticsView.swift +0 -94
  223. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/SettingsView.swift +0 -76
  224. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/TasksHomeView.swift +0 -364
  225. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Support/AppModel.swift +0 -324
  226. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.pbxproj +0 -439
  227. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.xcworkspace/contents.xcworkspacedata +0 -7
  228. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/xcshareddata/xcschemes/TodoOrbit.xcscheme +0 -89
  229. package/examples/todo-orbit/generated/ios/Todo Orbit/project.yml +0 -32
  230. package/examples/todo-orbit/generated/web/Todo Orbit/index.html +0 -16
  231. package/examples/todo-orbit/generated/web/Todo Orbit/package-lock.json +0 -1087
  232. package/examples/todo-orbit/generated/web/Todo Orbit/package.json +0 -24
  233. package/examples/todo-orbit/generated/web/Todo Orbit/src/App.tsx +0 -2167
  234. package/examples/todo-orbit/generated/web/Todo Orbit/src/main.tsx +0 -13
  235. package/examples/todo-orbit/generated/web/Todo Orbit/src/styles.css +0 -926
  236. package/examples/todo-orbit/generated/web/Todo Orbit/tsconfig.json +0 -19
  237. package/examples/todo-orbit/generated/web/Todo Orbit/vite.config.ts +0 -6
@@ -0,0 +1,507 @@
1
+ #!/usr/bin/env npx tsx
2
+ /**
3
+ * Takes screenshots of all generated targets across all example projects.
4
+ * Outputs to artifacts/<project>/screenshots/<platform>-<screen>.png
5
+ *
6
+ * Usage:
7
+ * npx tsx scripts/take-all-screenshots.ts # per-screen mode (manual nav)
8
+ * npx tsx scripts/take-all-screenshots.ts --batch # batch mode (build once, capture many)
9
+ *
10
+ * Requires: puppeteer, running Android emulator, booted iOS simulator.
11
+ */
12
+
13
+ import { spawn } from "node:child_process";
14
+ import { exec as execCb } from "node:child_process";
15
+ import { promisify } from "node:util";
16
+ import { mkdirSync, existsSync, readFileSync, writeFileSync, readdirSync } from "node:fs";
17
+ import { join, resolve } from "node:path";
18
+ import type { ChildProcess } from "node:child_process";
19
+
20
+ // Import helpers from mcp-server modules (per-screen mode)
21
+ import {
22
+ findAdb,
23
+ getConnectedEmulator,
24
+ adbShell,
25
+ extractAppInfo as extractAndroidAppInfo,
26
+ buildApk,
27
+ navigateByTaps,
28
+ captureScreenshot as captureAndroidScreenshot,
29
+ cleanEmulatorStorage,
30
+ } from "../mcp-server/screenshot-android.js";
31
+ import {
32
+ type IOSAppInfo,
33
+ extractAppInfo as extractIOSAppInfo,
34
+ findSimulator,
35
+ buildApp as buildIOSApp,
36
+ findAppBundle,
37
+ installAndLaunch as installAndLaunchIOS,
38
+ captureScreenshot as captureIOSScreenshot,
39
+ generateUITestTargetYml,
40
+ insertUITestTarget,
41
+ ensureInfoPlistFlag,
42
+ } from "../mcp-server/screenshot-ios.js";
43
+
44
+ // Import batch functions
45
+ import { takeScreenshotBatch } from "../mcp-server/screenshot.js";
46
+ import { takeAndroidScreenshotBatch } from "../mcp-server/screenshot-android.js";
47
+ import { takeIOSScreenshotBatch } from "../mcp-server/screenshot-ios.js";
48
+
49
+ const exec = promisify(execCb);
50
+
51
+ const ROOT = resolve(import.meta.dirname!, "..");
52
+ const ARTIFACTS = join(ROOT, "artifacts");
53
+ const BATCH_MODE = process.argv.includes("--batch");
54
+ const PLATFORM_FILTER = (() => {
55
+ const idx = process.argv.indexOf("--platform");
56
+ return idx >= 0 ? process.argv[idx + 1]?.toLowerCase() : null;
57
+ })();
58
+
59
+ // ── Project definitions ──────────────────────────────────────────────
60
+
61
+ interface WebScreen { name: string; route: string }
62
+ interface NativeScreen { name: string; route?: string; nav?: string[] }
63
+
64
+ interface ProjectDef {
65
+ name: string;
66
+ web?: { dir: string; screens: WebScreen[] };
67
+ android?: { dir: string; screens: NativeScreen[] };
68
+ ios?: { dir: string; screens: NativeScreen[] };
69
+ }
70
+
71
+ const PROJECTS: ProjectDef[] = [
72
+ {
73
+ name: "social-app",
74
+ web: {
75
+ dir: "examples/social-app/generated/web/social-app",
76
+ screens: [
77
+ { name: "home", route: "/home" },
78
+ { name: "discover", route: "/discover" },
79
+ { name: "notifications", route: "/notifications" },
80
+ { name: "messages", route: "/messages" },
81
+ { name: "profile", route: "/profile" },
82
+ { name: "settings", route: "/settings" },
83
+ ],
84
+ },
85
+ android: {
86
+ dir: "examples/social-app/generated/android/social-app",
87
+ screens: [
88
+ { name: "home", route: "socialapp://home" },
89
+ { name: "discover", route: "socialapp://discover" },
90
+ { name: "notifications", route: "socialapp://notifications" },
91
+ { name: "messages", route: "socialapp://messages" },
92
+ { name: "profile", route: "socialapp://profile" },
93
+ ],
94
+ },
95
+ },
96
+ {
97
+ name: "todo-orbit",
98
+ web: {
99
+ dir: "examples/todo-orbit/generated/web/Todo Orbit",
100
+ screens: [
101
+ { name: "home", route: "/" },
102
+ { name: "analytics", route: "/analytics" },
103
+ { name: "settings", route: "/settings" },
104
+ ],
105
+ },
106
+ android: {
107
+ dir: "examples/todo-orbit/generated/android/Todo Orbit",
108
+ screens: [
109
+ { name: "home" },
110
+ { name: "analytics", nav: ["Analytics"] },
111
+ { name: "settings", nav: ["Settings"] },
112
+ ],
113
+ },
114
+ ios: {
115
+ dir: "examples/todo-orbit/generated/ios/Todo Orbit",
116
+ screens: [
117
+ { name: "home" },
118
+ { name: "analytics", nav: ["Analytics"] },
119
+ { name: "settings", nav: ["Settings"] },
120
+ ],
121
+ },
122
+ },
123
+ {
124
+ name: "taskflow",
125
+ web: {
126
+ dir: "examples/taskflow/generated/web/TaskFlow",
127
+ screens: [
128
+ { name: "home", route: "/tasks" },
129
+ { name: "projects", route: "/projects" },
130
+ { name: "calendar", route: "/calendar" },
131
+ { name: "settings", route: "/settings" },
132
+ { name: "profile", route: "/profile" },
133
+ ],
134
+ },
135
+ android: {
136
+ dir: "examples/taskflow/generated/android/TaskFlow",
137
+ screens: [
138
+ { name: "home" },
139
+ { name: "projects", nav: ["Projects"] },
140
+ { name: "settings", nav: ["Settings"] },
141
+ ],
142
+ },
143
+ ios: {
144
+ dir: "examples/taskflow/generated/ios/TaskFlow",
145
+ screens: [
146
+ { name: "home" },
147
+ { name: "projects", nav: ["Projects"] },
148
+ { name: "calendar", nav: ["Calendar"] },
149
+ { name: "settings", nav: ["Settings"] },
150
+ ],
151
+ },
152
+ },
153
+ ];
154
+
155
+ // ── Utilities ────────────────────────────────────────────────────────
156
+
157
+ function log(msg: string) { console.log(`\x1b[36m▸\x1b[0m ${msg}`); }
158
+ function logOk(msg: string) { console.log(`\x1b[32m✔\x1b[0m ${msg}`); }
159
+ function logErr(msg: string) { console.error(`\x1b[31m✖\x1b[0m ${msg}`); }
160
+ function sleep(ms: number) { return new Promise((r) => setTimeout(r, ms)); }
161
+
162
+ function saveResultScreenshots(result: any, outDir: string, platform: string) {
163
+ mkdirSync(outDir, { recursive: true });
164
+ if (result.isError) {
165
+ logErr(` ${platform}: ${result.content?.[0]?.text ?? "unknown error"}`);
166
+ return;
167
+ }
168
+ for (const item of result.content) {
169
+ if (item.type === "image" && item.data) {
170
+ // Next text item has metadata with screen name
171
+ const idx = result.content.indexOf(item);
172
+ const meta = result.content[idx + 1];
173
+ let screenName = "unknown";
174
+ if (meta?.type === "text") {
175
+ try { screenName = JSON.parse(meta.text).screen; } catch { /* ignore */ }
176
+ }
177
+ const outPath = join(outDir, `${platform}-${screenName}.png`);
178
+ writeFileSync(outPath, Buffer.from(item.data, "base64"));
179
+ logOk(` ${platform}-${screenName}.png`);
180
+ }
181
+ }
182
+ }
183
+
184
+ // ══════════════════════════════════════════════════════════════════════
185
+ // BATCH MODE — uses takeScreenshotBatch / takeAndroidScreenshotBatch / takeIOSScreenshotBatch
186
+ // ══════════════════════════════════════════════════════════════════════
187
+
188
+ async function runBatchMode() {
189
+ for (const project of PROJECTS) {
190
+ console.log(`\n\x1b[1m=== ${project.name} (batch) ===\x1b[0m\n`);
191
+ const outDir = join(ARTIFACTS, project.name, "screenshots");
192
+
193
+ if (project.ios && (!PLATFORM_FILTER || PLATFORM_FILTER === "ios")) {
194
+ try {
195
+ log(`iOS batch: ${project.ios.screens.length} screens...`);
196
+ const result = await takeIOSScreenshotBatch(join(ROOT, "examples", project.name), {
197
+ captures: project.ios.screens.map((s) => ({ screen: s.name, nav: s.nav, wait_for: 5000 })),
198
+ project_dir: join(ROOT, project.ios.dir),
199
+ });
200
+ saveResultScreenshots(result, outDir, "ios");
201
+ } catch (err: any) { logErr(`iOS batch failed for ${project.name}: ${err.message}`); }
202
+ }
203
+
204
+ if (project.android && (!PLATFORM_FILTER || PLATFORM_FILTER === "android")) {
205
+ try {
206
+ log(`Android batch: ${project.android.screens.length} screens...`);
207
+ const result = await takeAndroidScreenshotBatch(join(ROOT, "examples", project.name), {
208
+ captures: project.android.screens.map((s) => ({ screen: s.name, route: s.route, nav: s.nav, wait_for: 8000 })),
209
+ project_dir: join(ROOT, project.android.dir),
210
+ });
211
+ saveResultScreenshots(result, outDir, "android");
212
+ } catch (err: any) { logErr(`Android batch failed for ${project.name}: ${err.message}`); }
213
+ }
214
+
215
+ if (project.web && (!PLATFORM_FILTER || PLATFORM_FILTER === "web")) {
216
+ try {
217
+ const openuispecDir = join(ROOT, "examples", project.name);
218
+ log(`Web batch: ${project.web.screens.length} screens...`);
219
+ const result = await takeScreenshotBatch(openuispecDir, {
220
+ captures: project.web.screens.map((s) => ({ screen: s.name, route: s.route, wait_for: 3000 })),
221
+ });
222
+ saveResultScreenshots(result, outDir, "web");
223
+ } catch (err: any) { logErr(`Web batch failed for ${project.name}: ${err.message}`); }
224
+ }
225
+ }
226
+ }
227
+
228
+ // ══════════════════════════════════════════════════════════════════════
229
+ // PER-SCREEN MODE — manual vite + puppeteer for web, adb for android, simctl + XCUITest for iOS
230
+ // ══════════════════════════════════════════════════════════════════════
231
+
232
+ async function startViteServer(dir: string): Promise<{ proc: ChildProcess; url: string }> {
233
+ return new Promise((resolve, reject) => {
234
+ const proc = spawn("npx", ["vite", "--port", "0"], {
235
+ cwd: dir,
236
+ stdio: ["pipe", "pipe", "pipe"],
237
+ env: { ...process.env, BROWSER: "none" },
238
+ });
239
+
240
+ let output = "";
241
+ const timeout = setTimeout(() => {
242
+ proc.kill();
243
+ reject(new Error(`Vite server timed out. Output: ${output}`));
244
+ }, 30_000);
245
+
246
+ const onData = (data: Buffer) => {
247
+ output += data.toString();
248
+ const match = output.match(/Local:\s+(https?:\/\/[^\s]+)/);
249
+ if (match) {
250
+ clearTimeout(timeout);
251
+ proc.stdout?.removeListener("data", onData);
252
+ proc.stderr?.removeListener("data", onData);
253
+ resolve({ proc, url: match[1].replace(/\/+$/, "") });
254
+ }
255
+ };
256
+
257
+ proc.stdout?.on("data", onData);
258
+ proc.stderr?.on("data", onData);
259
+ proc.on("error", (err) => { clearTimeout(timeout); reject(err); });
260
+ });
261
+ }
262
+
263
+ async function takeWebScreenshots(project: string, def: NonNullable<ProjectDef["web"]>) {
264
+ const outDir = join(ARTIFACTS, project, "screenshots");
265
+ mkdirSync(outDir, { recursive: true });
266
+
267
+ log(`Starting web server for ${project}...`);
268
+ const { proc, url } = await startViteServer(join(ROOT, def.dir));
269
+
270
+ try {
271
+ const puppeteer = await import("puppeteer");
272
+ const browser = await puppeteer.default.launch({ headless: "shell" });
273
+ try {
274
+ const page = await browser.newPage();
275
+ await page.setViewport({ width: 1280, height: 800 });
276
+ for (const screen of def.screens) {
277
+ const fullUrl = `${url}${screen.route}`;
278
+ log(` web/${screen.name}: ${fullUrl}`);
279
+ await page.goto(fullUrl, { waitUntil: "networkidle0", timeout: 15_000 });
280
+ try {
281
+ await page.waitForFunction(
282
+ () => (document.getElementById("root")?.children.length ?? 0) > 0,
283
+ { timeout: 8_000 },
284
+ );
285
+ } catch { /* app may not use #root */ }
286
+ await sleep(3000);
287
+ await page.screenshot({ path: join(outDir, `web-${screen.name}.png`), fullPage: false });
288
+ logOk(` web-${screen.name}.png`);
289
+ }
290
+ } finally {
291
+ await browser.close();
292
+ }
293
+ } finally {
294
+ proc.kill();
295
+ }
296
+ }
297
+
298
+ async function takeAndroidScreenshots(project: string, def: NonNullable<ProjectDef["android"]>) {
299
+ const outDir = join(ARTIFACTS, project, "screenshots");
300
+ mkdirSync(outDir, { recursive: true });
301
+
302
+ const androidDir = join(ROOT, def.dir);
303
+ const adb = findAdb();
304
+ const serial = await getConnectedEmulator(adb);
305
+
306
+ log(`Cleaning emulator storage...`);
307
+ await cleanEmulatorStorage(adb, serial);
308
+
309
+ const appInfo = extractAndroidAppInfo(androidDir);
310
+ log(`Building Android APK for ${project}...`);
311
+ const apkPath = await buildApk(androidDir, appInfo.moduleName);
312
+
313
+ log(`Installing on emulator ${serial}...`);
314
+ await exec(`${adb} -s ${serial} install -r "${apkPath}"`, { timeout: 60_000 });
315
+
316
+ for (const screen of def.screens) {
317
+ log(` android/${screen.name}...`);
318
+
319
+ await adbShell(adb, serial, `am force-stop ${appInfo.applicationId}`);
320
+ try { await adbShell(adb, serial, `pm clear ${appInfo.applicationId}`); } catch { /* ignore */ }
321
+ await sleep(500);
322
+
323
+ if (screen.route) {
324
+ // Deep link launch
325
+ await adbShell(adb, serial,
326
+ `am start -W -a android.intent.action.VIEW -d '${screen.route}' ` +
327
+ `${appInfo.applicationId}/${appInfo.launchActivity}`);
328
+ } else {
329
+ // Normal launch + optional nav taps
330
+ await adbShell(adb, serial,
331
+ `am start -W -n ${appInfo.applicationId}/${appInfo.launchActivity}`);
332
+ }
333
+ await sleep(5000);
334
+
335
+ if (!screen.route && screen.nav && screen.nav.length > 0) {
336
+ try {
337
+ await navigateByTaps(adb, serial, screen.nav);
338
+ } catch (err: any) {
339
+ logErr(` Nav failed: ${err.message}`);
340
+ }
341
+ }
342
+
343
+ const outPath = join(outDir, `android-${screen.name}.png`);
344
+ await captureAndroidScreenshot(adb, serial, outPath);
345
+ logOk(` android-${screen.name}.png`);
346
+ }
347
+ }
348
+
349
+ async function takeIOSScreenshots(project: string, def: NonNullable<ProjectDef["ios"]>) {
350
+ const outDir = join(ARTIFACTS, project, "screenshots");
351
+ mkdirSync(outDir, { recursive: true });
352
+
353
+ const iosDir = join(ROOT, def.dir);
354
+ const appInfo = extractIOSAppInfo(iosDir);
355
+ const sim = findSimulator();
356
+ const simUdid = sim.udid;
357
+
358
+ log(`Building iOS app for ${project} (scheme: ${appInfo.schemeName})...`);
359
+ const appBundlePath = await buildIOSApp(iosDir, appInfo, simUdid);
360
+ log(`Installing on simulator...`);
361
+ await installAndLaunchIOS(simUdid, appBundlePath, appInfo.bundleId);
362
+
363
+ const homeScreen = def.screens.find((s) => !s.nav || s.nav.length === 0);
364
+ if (homeScreen) {
365
+ log(` ios/${homeScreen.name} (launch screenshot)...`);
366
+ await sleep(5000);
367
+ await captureIOSScreenshot(simUdid, join(outDir, `ios-${homeScreen.name}.png`));
368
+ logOk(` ios-${homeScreen.name}.png`);
369
+ }
370
+
371
+ const navScreens = def.screens.filter((s) => s.nav && s.nav.length > 0);
372
+ if (navScreens.length === 0) return;
373
+
374
+ log(` Generating XCUITest for ${navScreens.length} nav screens...`);
375
+
376
+ const uitestDir = join(iosDir, ".screenshot-uitest");
377
+ const sourcesDir = join(uitestDir, "Sources");
378
+ mkdirSync(sourcesDir, { recursive: true });
379
+
380
+ const testCases = navScreens.map((screen, i) => {
381
+ const taps = (screen.nav ?? []).map((step, j) => {
382
+ const escaped = step.replace(/\\/g, "\\\\").replace(/"/g, '\\"');
383
+ return `
384
+ let target_${i}_${j} = app.descendants(matching: .any).matching(NSPredicate(format: "label ==[c] %@ OR title ==[c] %@", "${escaped}", "${escaped}")).firstMatch
385
+ if target_${i}_${j}.waitForExistence(timeout: 5) {
386
+ target_${i}_${j}.tap()
387
+ Thread.sleep(forTimeInterval: 0.8)
388
+ }`;
389
+ }).join("\n");
390
+
391
+ const outputPath = join(outDir, `ios-${screen.name}.png`).replace(/"/g, '\\"');
392
+ return `
393
+ func test_${String(i + 1).padStart(2, "0")}_${screen.name}() {
394
+ let app = XCUIApplication()
395
+ app.launchArguments = ["-AppleLanguages", "(en)"]
396
+ app.launch()
397
+ Thread.sleep(forTimeInterval: 2.0)
398
+ ${taps}
399
+ Thread.sleep(forTimeInterval: 0.5)
400
+ let screenshot = XCUIScreen.main.screenshot()
401
+ try! screenshot.pngRepresentation.write(to: URL(fileURLWithPath: "${outputPath}"))
402
+ }`;
403
+ }).join("\n");
404
+
405
+ writeFileSync(join(sourcesDir, "ScreenshotUITest.swift"),
406
+ `import XCTest\n\nfinal class ScreenshotUITest: XCTestCase {\n${testCases}\n}\n`);
407
+
408
+ const UITEST_TARGET = "ScreenshotUITests";
409
+ const hasXcodegen = existsSync(join(iosDir, "project.yml"));
410
+ const projectYmlPath = join(iosDir, "project.yml");
411
+ let originalProjectYml: string | null = null;
412
+ const buildDir = join(iosDir, ".build", "screenshot");
413
+
414
+ if (hasXcodegen) {
415
+ originalProjectYml = readFileSync(projectYmlPath, "utf-8");
416
+ let modifiedYml = ensureInfoPlistFlag(originalProjectYml);
417
+ modifiedYml = insertUITestTarget(modifiedYml, generateUITestTargetYml(appInfo, ".screenshot-uitest/Sources", true));
418
+ writeFileSync(projectYmlPath, modifiedYml);
419
+ await exec(`xcodegen generate`, { cwd: iosDir, timeout: 30_000 });
420
+ } else {
421
+ writeFileSync(join(uitestDir, "project.yml"), `name: ${UITEST_TARGET}
422
+ targets:
423
+ ${UITEST_TARGET}:
424
+ type: bundle.ui-testing
425
+ platform: iOS
426
+ deploymentTarget: "${appInfo.deploymentTarget}"
427
+ sources:
428
+ - path: Sources
429
+ settings:
430
+ base:
431
+ TEST_TARGET_NAME: ${appInfo.schemeName}
432
+ PRODUCT_BUNDLE_IDENTIFIER: ${appInfo.bundleId}.uitests
433
+ GENERATE_INFOPLIST_FILE: YES
434
+ `);
435
+ await exec(`xcodegen generate`, { cwd: uitestDir, timeout: 30_000 });
436
+ }
437
+
438
+ const testProjectFlag = hasXcodegen
439
+ ? (appInfo.xcodeproj ? `-project "${join(iosDir, appInfo.xcodeproj)}"` : "")
440
+ : `-project "${join(uitestDir, `${UITEST_TARGET}.xcodeproj`)}"`;
441
+ const testCwd = hasXcodegen ? iosDir : uitestDir;
442
+
443
+ try {
444
+ log(` Running XCUITest to capture ${navScreens.length} screens...`);
445
+ await exec(
446
+ `xcodebuild test ${testProjectFlag} -scheme "${UITEST_TARGET}" -destination "id=${simUdid}" -derivedDataPath "${buildDir}" -only-testing:${UITEST_TARGET}/ScreenshotUITest 2>&1`,
447
+ { cwd: testCwd, timeout: 300_000 },
448
+ );
449
+ } catch {
450
+ const missing = navScreens.filter((s) => !existsSync(join(outDir, `ios-${s.name}.png`)));
451
+ if (missing.length > 0) {
452
+ logErr(` XCUITest failed for: ${missing.map((s) => s.name).join(", ")}`);
453
+ }
454
+ } finally {
455
+ if (originalProjectYml) {
456
+ writeFileSync(projectYmlPath, originalProjectYml);
457
+ try { await exec(`xcodegen generate`, { cwd: iosDir, timeout: 30_000 }); } catch { /* best effort */ }
458
+ }
459
+ }
460
+
461
+ for (const screen of navScreens) {
462
+ if (existsSync(join(outDir, `ios-${screen.name}.png`))) {
463
+ logOk(` ios-${screen.name}.png`);
464
+ }
465
+ }
466
+ }
467
+
468
+ async function runPerScreenMode() {
469
+ for (const project of PROJECTS) {
470
+ console.log(`\n\x1b[1m=== ${project.name} ===\x1b[0m\n`);
471
+
472
+ if (project.ios && (!PLATFORM_FILTER || PLATFORM_FILTER === "ios")) {
473
+ try { await takeIOSScreenshots(project.name, project.ios); }
474
+ catch (err: any) { logErr(`iOS screenshots failed for ${project.name}: ${err.message}`); }
475
+ }
476
+
477
+ if (project.android && (!PLATFORM_FILTER || PLATFORM_FILTER === "android")) {
478
+ try { await takeAndroidScreenshots(project.name, project.android); }
479
+ catch (err: any) { logErr(`Android screenshots failed for ${project.name}: ${err.message}`); }
480
+ }
481
+
482
+ if (project.web && (!PLATFORM_FILTER || PLATFORM_FILTER === "web")) {
483
+ try { await takeWebScreenshots(project.name, project.web); }
484
+ catch (err: any) { logErr(`Web screenshots failed for ${project.name}: ${err.message}`); }
485
+ }
486
+ }
487
+ }
488
+
489
+ // ── Main ─────────────────────────────────────────────────────────────
490
+
491
+ async function main() {
492
+ const mode = BATCH_MODE ? "batch" : "per-screen";
493
+ console.log(`\nTaking screenshots of all generated targets (${mode} mode)\n`);
494
+
495
+ if (BATCH_MODE) {
496
+ await runBatchMode();
497
+ } else {
498
+ await runPerScreenMode();
499
+ }
500
+
501
+ console.log("\n\x1b[32mDone! Screenshots saved to artifacts/\x1b[0m\n");
502
+ }
503
+
504
+ main().catch((err) => {
505
+ console.error(err);
506
+ process.exit(1);
507
+ });
@@ -90,6 +90,19 @@ generation:
90
90
  ios: { language: swift, framework: swiftui }
91
91
  android: { language: kotlin, framework: compose }
92
92
  web: { language: typescript, framework: react }
93
+ # shared: # optional: cross-platform shared code layers
94
+ # mobile_common:
95
+ # platforms: [ios, android]
96
+ # language: kotlin
97
+ # root: "../shared"
98
+ # scope: "Business logic, data models, repositories, view models. No UI."
99
+ # paths:
100
+ # domain: "commonMain/domain/"
101
+ # structure: # optional: per-target directory structure (overrides heuristics)
102
+ # ios:
103
+ # root: "../shared"
104
+ # scope: "Pure SwiftUI views and navigation."
105
+ # paths: { ui: "iosApp/ui/" }
93
106
  ```
94
107
 
95
108
  ---
package/status/index.ts CHANGED
@@ -10,14 +10,19 @@
10
10
  import { existsSync } from "node:fs";
11
11
  import {
12
12
  computeDrift,
13
+ computeSharedDrift,
13
14
  discoverTargets,
14
15
  explainDrift,
15
16
  findProjectDir,
16
17
  formatBaseline,
18
+ hasDriftChanges,
17
19
  readOutputDirs,
18
20
  readProjectName,
21
+ readSharedLayers,
22
+ readSharedLayerState,
19
23
  resolveOutputDir,
20
24
  stateFilePath,
25
+ type SharedLayerConfig,
21
26
  type StateFile,
22
27
  } from "../drift/index.js";
23
28
 
@@ -45,9 +50,21 @@ interface TargetStatus {
45
50
  note?: string;
46
51
  }
47
52
 
53
+ interface SharedLayerStatus {
54
+ name: string;
55
+ platforms: string[];
56
+ root: string;
57
+ snapshot: boolean;
58
+ snapshot_at: string | null;
59
+ generated_by_target: string | null;
60
+ has_drift: boolean;
61
+ status: "up to date" | "behind" | "needs generation";
62
+ }
63
+
48
64
  export interface StatusResult {
49
65
  project: string;
50
66
  targets: TargetStatus[];
67
+ shared_layers?: SharedLayerStatus[];
51
68
  }
52
69
 
53
70
  function configuredTargets(projectDir: string): string[] {
@@ -100,10 +117,10 @@ function buildTargetStatus(cwd: string, projectDir: string, projectName: string,
100
117
  explain_available: false,
101
118
  status: outputExists ? "needs baseline" : "needs generation",
102
119
  recommended_next_step: outputExists
103
- ? `Run \`openuispec drift --snapshot --target ${target}\` after reviewing the generated output.`
120
+ ? `Review the generated output, then run \`openuispec drift --snapshot --target ${target}\` to create the baseline.`
104
121
  : `Run code generation for "${target}", then \`openuispec prepare --target ${target}\` to build the target work bundle.`,
105
122
  note: outputExists
106
- ? "No snapshot found for this target."
123
+ ? "Baseline pending generated code exists but user has not yet confirmed it with a snapshot."
107
124
  : `Output directory not found. Run code generation for "${target}" first.`,
108
125
  };
109
126
  }
@@ -141,6 +158,38 @@ function buildTargetStatus(cwd: string, projectDir: string, projectName: string,
141
158
  };
142
159
  }
143
160
 
161
+ function buildSharedLayerStatus(projectDir: string, layer: SharedLayerConfig): SharedLayerStatus {
162
+ const state = readSharedLayerState(layer);
163
+ if (!state) {
164
+ return {
165
+ name: layer.name,
166
+ platforms: layer.platforms,
167
+ root: layer.root,
168
+ snapshot: false,
169
+ snapshot_at: null,
170
+ generated_by_target: null,
171
+ has_drift: false,
172
+ status: "needs generation",
173
+ };
174
+ }
175
+
176
+ let hasDrift = false;
177
+ if (layer.tracks.length > 0) {
178
+ hasDrift = hasDriftChanges(computeSharedDrift(projectDir, layer).drift);
179
+ }
180
+
181
+ return {
182
+ name: layer.name,
183
+ platforms: layer.platforms,
184
+ root: layer.root,
185
+ snapshot: true,
186
+ snapshot_at: state.snapshot_at,
187
+ generated_by_target: state.generated_by_target,
188
+ has_drift: hasDrift,
189
+ status: hasDrift ? "behind" : "up to date",
190
+ };
191
+ }
192
+
144
193
  export function buildStatusResult(cwd: string = process.cwd()): StatusResult {
145
194
  const projectDir = findProjectDir(cwd);
146
195
  const projectName = readProjectName(projectDir);
@@ -148,9 +197,15 @@ export function buildStatusResult(cwd: string = process.cwd()): StatusResult {
148
197
  buildTargetStatus(cwd, projectDir, projectName, target)
149
198
  );
150
199
 
200
+ const sharedLayers = readSharedLayers(projectDir);
201
+ const sharedLayerStatuses = sharedLayers.length > 0
202
+ ? sharedLayers.map((layer) => buildSharedLayerStatus(projectDir, layer))
203
+ : undefined;
204
+
151
205
  return {
152
206
  project: projectName,
153
207
  targets,
208
+ ...(sharedLayerStatuses ? { shared_layers: sharedLayerStatuses } : {}),
154
209
  };
155
210
  }
156
211
 
@@ -182,6 +237,21 @@ function printReport(result: StatusResult): void {
182
237
  console.log(` next: ${target.recommended_next_step}`);
183
238
  console.log("");
184
239
  }
240
+
241
+ if (result.shared_layers && result.shared_layers.length > 0) {
242
+ console.log("Shared Layers");
243
+ console.log("─────────────");
244
+ for (const layer of result.shared_layers) {
245
+ console.log(`${layer.name} (${layer.platforms.join(", ")})`);
246
+ console.log(` root: ${layer.root}`);
247
+ console.log(` snapshot: ${layer.snapshot ? layer.snapshot_at : "missing"}`);
248
+ if (layer.generated_by_target) {
249
+ console.log(` generated by: ${layer.generated_by_target}`);
250
+ }
251
+ console.log(` status: ${layer.status}`);
252
+ console.log("");
253
+ }
254
+ }
185
255
  }
186
256
 
187
257
  export function runStatus(argv: string[]): void {
@@ -1,10 +0,0 @@
1
- {
2
- "mcpServers": {
3
- "openuispec": {
4
- "command": "openuispec",
5
- "args": [
6
- "mcp"
7
- ]
8
- }
9
- }
10
- }