openuispec 0.2.9 → 0.2.11

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 (232) hide show
  1. package/README.md +3 -1
  2. package/cli/index.ts +80 -2
  3. package/cli/init.ts +23 -8
  4. package/docs/cli.md +42 -7
  5. package/drift/index.ts +41 -15
  6. package/mcp-server/index.ts +247 -117
  7. package/mcp-server/screenshot-android.ts +185 -44
  8. package/mcp-server/screenshot-ios.ts +242 -30
  9. package/mcp-server/screenshot.ts +96 -1
  10. package/package.json +5 -2
  11. package/prepare/index.ts +16 -0
  12. package/scripts/take-all-screenshots.ts +507 -0
  13. package/status/index.ts +2 -2
  14. package/examples/social-app/.mcp.json +0 -10
  15. package/examples/social-app/AGENTS.md +0 -124
  16. package/examples/social-app/CLAUDE.md +0 -124
  17. package/examples/social-app/backend/.gitkeep +0 -1
  18. package/examples/social-app/generated/android/social-app/app/.paparazzi-hashes.json +0 -3
  19. package/examples/social-app/generated/android/social-app/app/build.gradle.kts +0 -94
  20. package/examples/social-app/generated/android/social-app/app/src/main/AndroidManifest.xml +0 -26
  21. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/AppContainer.kt +0 -20
  22. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/MainActivity.kt +0 -35
  23. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/SocialAppApplication.kt +0 -13
  24. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/MockData.kt +0 -98
  25. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/AppPreferences.kt +0 -19
  26. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/DataStorePreferencesRepository.kt +0 -68
  27. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/PreferencesRepository.kt +0 -15
  28. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/model/Models.kt +0 -34
  29. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/MainShell.kt +0 -390
  30. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/Components.kt +0 -234
  31. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/ContractPrimitives.kt +0 -641
  32. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/navigation/RootComponent.kt +0 -113
  33. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ChatDetailScreen.kt +0 -212
  34. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/CreatePostScreen.kt +0 -113
  35. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/DiscoverScreen.kt +0 -137
  36. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/EditProfileScreen.kt +0 -180
  37. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/HomeFeedScreen.kt +0 -169
  38. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/MessagesInboxScreen.kt +0 -85
  39. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/NotificationsScreen.kt +0 -74
  40. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/PostDetailScreen.kt +0 -293
  41. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ProfileSelfScreen.kt +0 -116
  42. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SearchResultsScreen.kt +0 -161
  43. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsScreen.kt +0 -164
  44. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsStore.kt +0 -95
  45. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/UserProfileScreen.kt +0 -123
  46. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Color.kt +0 -33
  47. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Shape.kt +0 -41
  48. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Spacing.kt +0 -20
  49. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Theme.kt +0 -82
  50. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Type.kt +0 -60
  51. package/examples/social-app/generated/android/social-app/app/src/main/res/drawable/ic_launcher_foreground.xml +0 -9
  52. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +0 -5
  53. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml +0 -5
  54. package/examples/social-app/generated/android/social-app/app/src/main/res/values/strings.xml +0 -91
  55. package/examples/social-app/generated/android/social-app/app/src/main/res/values/themes.xml +0 -10
  56. package/examples/social-app/generated/android/social-app/app/src/main/res/values-ru/strings.xml +0 -79
  57. package/examples/social-app/generated/android/social-app/app/src/main/res/values-uz/strings.xml +0 -79
  58. package/examples/social-app/generated/android/social-app/app/src/main/xml/AndroidManifest.xml +0 -23
  59. package/examples/social-app/generated/android/social-app/app/src/test/kotlin/com/social/app/screenshots/HomeFeedScreenshotTest.kt +0 -34
  60. package/examples/social-app/generated/android/social-app/build.gradle.kts +0 -7
  61. package/examples/social-app/generated/android/social-app/gradle/libs.versions.toml +0 -50
  62. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.jar +0 -0
  63. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.properties +0 -8
  64. package/examples/social-app/generated/android/social-app/gradle.properties +0 -11
  65. package/examples/social-app/generated/android/social-app/gradlew +0 -248
  66. package/examples/social-app/generated/android/social-app/settings.gradle.kts +0 -27
  67. package/examples/social-app/generated/web/social-app/index.html +0 -12
  68. package/examples/social-app/generated/web/social-app/package-lock.json +0 -2517
  69. package/examples/social-app/generated/web/social-app/package.json +0 -27
  70. package/examples/social-app/generated/web/social-app/src/app/App.tsx +0 -58
  71. package/examples/social-app/generated/web/social-app/src/components/Shell.tsx +0 -259
  72. package/examples/social-app/generated/web/social-app/src/components/cards.tsx +0 -317
  73. package/examples/social-app/generated/web/social-app/src/components/ui.tsx +0 -340
  74. package/examples/social-app/generated/web/social-app/src/flows/CreatePostFlow.tsx +0 -86
  75. package/examples/social-app/generated/web/social-app/src/i18n.tsx +0 -59
  76. package/examples/social-app/generated/web/social-app/src/lib/icons.tsx +0 -85
  77. package/examples/social-app/generated/web/social-app/src/lib/tokens.ts +0 -70
  78. package/examples/social-app/generated/web/social-app/src/lib/utils.ts +0 -97
  79. package/examples/social-app/generated/web/social-app/src/locales/en.json +0 -67
  80. package/examples/social-app/generated/web/social-app/src/locales/ru.json +0 -67
  81. package/examples/social-app/generated/web/social-app/src/locales/uz.json +0 -67
  82. package/examples/social-app/generated/web/social-app/src/main.tsx +0 -16
  83. package/examples/social-app/generated/web/social-app/src/screens/ChatDetailScreen.tsx +0 -90
  84. package/examples/social-app/generated/web/social-app/src/screens/DiscoverScreen.tsx +0 -86
  85. package/examples/social-app/generated/web/social-app/src/screens/EditProfileScreen.tsx +0 -57
  86. package/examples/social-app/generated/web/social-app/src/screens/HomeFeedScreen.tsx +0 -103
  87. package/examples/social-app/generated/web/social-app/src/screens/MessagesInboxScreen.tsx +0 -52
  88. package/examples/social-app/generated/web/social-app/src/screens/NotificationsScreen.tsx +0 -41
  89. package/examples/social-app/generated/web/social-app/src/screens/PostDetailScreen.tsx +0 -115
  90. package/examples/social-app/generated/web/social-app/src/screens/ProfileSelfScreen.tsx +0 -57
  91. package/examples/social-app/generated/web/social-app/src/screens/ProfileUserScreen.tsx +0 -76
  92. package/examples/social-app/generated/web/social-app/src/screens/SearchResultsScreen.tsx +0 -96
  93. package/examples/social-app/generated/web/social-app/src/screens/SettingsScreen.tsx +0 -79
  94. package/examples/social-app/generated/web/social-app/src/state/store.ts +0 -592
  95. package/examples/social-app/generated/web/social-app/src/styles.css +0 -125
  96. package/examples/social-app/generated/web/social-app/src/vite-env.d.ts +0 -1
  97. package/examples/social-app/generated/web/social-app/tsconfig.json +0 -22
  98. package/examples/social-app/generated/web/social-app/tsconfig.node.json +0 -13
  99. package/examples/social-app/generated/web/social-app/tsconfig.node.tsbuildinfo +0 -1
  100. package/examples/social-app/generated/web/social-app/tsconfig.tsbuildinfo +0 -1
  101. package/examples/social-app/generated/web/social-app/vite.config.d.ts +0 -2
  102. package/examples/social-app/generated/web/social-app/vite.config.js +0 -6
  103. package/examples/social-app/generated/web/social-app/vite.config.ts +0 -7
  104. package/examples/social-app/package.json +0 -13
  105. package/examples/social-app/take-web-screenshots.ts +0 -97
  106. package/examples/taskflow/.codex/config.toml +0 -4
  107. package/examples/taskflow/.mcp.json +0 -10
  108. package/examples/taskflow/AGENTS.md +0 -124
  109. package/examples/taskflow/CLAUDE.md +0 -124
  110. package/examples/taskflow/backend/.gitkeep +0 -1
  111. package/examples/taskflow/generated/android/TaskFlow/README.md +0 -43
  112. package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +0 -76
  113. package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +0 -1
  114. package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +0 -21
  115. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +0 -19
  116. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +0 -283
  117. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +0 -106
  118. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +0 -57
  119. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +0 -109
  120. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +0 -112
  121. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +0 -61
  122. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +0 -82
  123. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +0 -111
  124. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +0 -77
  125. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +0 -30
  126. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +0 -86
  127. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +0 -57
  128. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +0 -155
  129. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +0 -4
  130. package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +0 -5
  131. package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +0 -12
  132. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
  133. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +0 -7
  134. package/examples/taskflow/generated/android/TaskFlow/gradle.properties +0 -4
  135. package/examples/taskflow/generated/android/TaskFlow/gradlew +0 -18
  136. package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +0 -12
  137. package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +0 -18
  138. package/examples/taskflow/generated/ios/TaskFlow/README.md +0 -21
  139. package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +0 -115
  140. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +0 -24
  141. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +0 -150
  142. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +0 -220
  143. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +0 -122
  144. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +0 -21
  145. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +0 -201
  146. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +0 -48
  147. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +0 -59
  148. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +0 -63
  149. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +0 -85
  150. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +0 -219
  151. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +0 -320
  152. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +0 -41
  153. package/examples/taskflow/generated/ios/TaskFlow/project.yml +0 -26
  154. package/examples/taskflow/generated/web/TaskFlow/README.md +0 -19
  155. package/examples/taskflow/generated/web/TaskFlow/index.html +0 -12
  156. package/examples/taskflow/generated/web/TaskFlow/package-lock.json +0 -1908
  157. package/examples/taskflow/generated/web/TaskFlow/package.json +0 -24
  158. package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +0 -58
  159. package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +0 -55
  160. package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +0 -82
  161. package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +0 -191
  162. package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +0 -41
  163. package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +0 -131
  164. package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +0 -25
  165. package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +0 -39
  166. package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +0 -111
  167. package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +0 -13
  168. package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +0 -111
  169. package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +0 -82
  170. package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +0 -132
  171. package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +0 -105
  172. package/examples/taskflow/generated/web/TaskFlow/src/store.ts +0 -216
  173. package/examples/taskflow/generated/web/TaskFlow/src/styles.css +0 -617
  174. package/examples/taskflow/generated/web/TaskFlow/src/types.ts +0 -64
  175. package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +0 -78
  176. package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +0 -21
  177. package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +0 -6
  178. package/examples/todo-orbit/.codex/config.toml +0 -4
  179. package/examples/todo-orbit/.mcp.json +0 -10
  180. package/examples/todo-orbit/AGENTS.md +0 -124
  181. package/examples/todo-orbit/CLAUDE.md +0 -124
  182. package/examples/todo-orbit/backend/.gitkeep +0 -1
  183. package/examples/todo-orbit/generated/android/Todo Orbit/README.md +0 -14
  184. package/examples/todo-orbit/generated/android/Todo Orbit/app/build.gradle.kts +0 -58
  185. package/examples/todo-orbit/generated/android/Todo Orbit/app/proguard-rules.pro +0 -1
  186. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/AndroidManifest.xml +0 -20
  187. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/MainActivity.kt +0 -14
  188. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/TodoOrbitApp.kt +0 -345
  189. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/AppLogic.kt +0 -231
  190. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Models.kt +0 -169
  191. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Strings.kt +0 -8
  192. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/components/CommonComponents.kt +0 -236
  193. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/AnalyticsScreen.kt +0 -193
  194. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/SettingsScreen.kt +0 -102
  195. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/TasksScreen.kt +0 -347
  196. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/sheets/EditorSheets.kt +0 -347
  197. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/theme/TodoOrbitTheme.kt +0 -59
  198. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values/strings.xml +0 -149
  199. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values-ru/strings.xml +0 -155
  200. package/examples/todo-orbit/generated/android/Todo Orbit/build.gradle.kts +0 -4
  201. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.jar +0 -0
  202. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.properties +0 -7
  203. package/examples/todo-orbit/generated/android/Todo Orbit/gradle.properties +0 -4
  204. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew +0 -248
  205. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew.bat +0 -93
  206. package/examples/todo-orbit/generated/android/Todo Orbit/settings.gradle.kts +0 -18
  207. package/examples/todo-orbit/generated/ios/Todo Orbit/.screenshot-uitest/Sources/ScreenshotUITest.swift +0 -36
  208. package/examples/todo-orbit/generated/ios/Todo Orbit/README.md +0 -29
  209. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/en.lproj/Localizable.strings +0 -119
  210. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/ru.lproj/Localizable.strings +0 -119
  211. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/App/TodoOrbitApp.swift +0 -50
  212. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/OrbitChrome.swift +0 -204
  213. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/SchedulePreviewView.swift +0 -126
  214. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/TrendChartView.swift +0 -70
  215. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/RecurringRuleSheet.swift +0 -126
  216. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/TaskEditorSheet.swift +0 -61
  217. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Models/DomainModels.swift +0 -238
  218. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/AnalyticsView.swift +0 -94
  219. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/SettingsView.swift +0 -76
  220. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/TasksHomeView.swift +0 -364
  221. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Support/AppModel.swift +0 -324
  222. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.pbxproj +0 -400
  223. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.xcworkspace/contents.xcworkspacedata +0 -7
  224. package/examples/todo-orbit/generated/ios/Todo Orbit/project.yml +0 -25
  225. package/examples/todo-orbit/generated/web/Todo Orbit/index.html +0 -16
  226. package/examples/todo-orbit/generated/web/Todo Orbit/package-lock.json +0 -1087
  227. package/examples/todo-orbit/generated/web/Todo Orbit/package.json +0 -24
  228. package/examples/todo-orbit/generated/web/Todo Orbit/src/App.tsx +0 -2167
  229. package/examples/todo-orbit/generated/web/Todo Orbit/src/main.tsx +0 -13
  230. package/examples/todo-orbit/generated/web/Todo Orbit/src/styles.css +0 -926
  231. package/examples/todo-orbit/generated/web/Todo Orbit/tsconfig.json +0 -19
  232. package/examples/todo-orbit/generated/web/Todo Orbit/vite.config.ts +0 -6
@@ -17,6 +17,7 @@ import { findProjectDir } from "../drift/index.js";
17
17
  export interface ScreenshotOptions {
18
18
  route: string;
19
19
  viewport?: { width: number; height: number };
20
+ scale?: number;
20
21
  theme?: "light" | "dark";
21
22
  wait_for?: number;
22
23
  full_page?: boolean;
@@ -177,6 +178,7 @@ export async function takeScreenshot(
177
178
  const {
178
179
  route = "/",
179
180
  viewport = { width: 1280, height: 800 },
181
+ scale = 2,
180
182
  theme,
181
183
  wait_for = 1000,
182
184
  full_page = false,
@@ -192,7 +194,11 @@ export async function takeScreenshot(
192
194
  // 2. Navigate
193
195
  const page = await browser.newPage();
194
196
  try {
195
- await page.setViewport({ width: viewport.width, height: viewport.height });
197
+ await page.setViewport({
198
+ width: viewport.width,
199
+ height: viewport.height,
200
+ deviceScaleFactor: scale,
201
+ });
196
202
 
197
203
  if (theme) {
198
204
  await page.emulateMediaFeatures([
@@ -245,6 +251,7 @@ export async function takeScreenshot(
245
251
  route,
246
252
  url: targetUrl,
247
253
  viewport,
254
+ scale,
248
255
  theme: theme ?? "default",
249
256
  full_page,
250
257
  selector: selector ?? null,
@@ -258,6 +265,94 @@ export async function takeScreenshot(
258
265
  }
259
266
  }
260
267
 
268
+ // ── batch types ──────────────────────────────────────────────────────
269
+
270
+ export interface WebBatchCapture {
271
+ screen: string;
272
+ route: string;
273
+ selector?: string;
274
+ full_page?: boolean;
275
+ wait_for?: number;
276
+ }
277
+
278
+ export interface WebScreenshotBatchOptions {
279
+ captures: WebBatchCapture[];
280
+ viewport?: { width: number; height: number };
281
+ scale?: number;
282
+ theme?: "light" | "dark";
283
+ output_dir?: string;
284
+ }
285
+
286
+ // ── batch screenshot ─────────────────────────────────────────────────
287
+
288
+ export async function takeScreenshotBatch(
289
+ projectCwd: string,
290
+ options: WebScreenshotBatchOptions,
291
+ ): Promise<ScreenshotResult> {
292
+ const { captures, viewport = { width: 1280, height: 800 }, scale = 2, theme, output_dir } = options;
293
+
294
+ if (captures.length === 0) {
295
+ return { content: [{ type: "text", text: "No web captures specified." }], isError: true };
296
+ }
297
+
298
+ const webDir = findWebAppDir(projectCwd);
299
+ const server = await startDevServer(webDir);
300
+ const browser = await getBrowser();
301
+ const page = await browser.newPage();
302
+
303
+ try {
304
+ await page.setViewport({
305
+ width: viewport.width,
306
+ height: viewport.height,
307
+ deviceScaleFactor: scale,
308
+ });
309
+ if (theme) {
310
+ await page.emulateMediaFeatures([{ name: "prefers-color-scheme", value: theme }]);
311
+ }
312
+
313
+ const base = server.url.replace(/\/+$/, "");
314
+ const themeLabel = theme ?? "default";
315
+ const snapshots: Array<{ screen: string; path: string; data: string }> = [];
316
+
317
+ for (const capture of captures) {
318
+ const targetUrl = `${base}${capture.route.startsWith("/") ? "" : "/"}${capture.route}`;
319
+ await page.goto(targetUrl, { waitUntil: "networkidle2", timeout: 30_000 });
320
+ await new Promise((r) => setTimeout(r, capture.wait_for ?? 1000));
321
+
322
+ let buffer: Buffer;
323
+ if (capture.selector) {
324
+ const el = await page.$(capture.selector);
325
+ buffer = el ? await el.screenshot({ type: "png" }) : await page.screenshot({ type: "png" });
326
+ } else {
327
+ buffer = await page.screenshot({ type: "png", fullPage: capture.full_page ?? false });
328
+ }
329
+
330
+ const filename = `${capture.screen}_${themeLabel}.png`;
331
+ let savedPath = filename;
332
+ if (output_dir) {
333
+ const outDir = resolve(webDir, output_dir);
334
+ mkdirSync(outDir, { recursive: true });
335
+ savedPath = join(outDir, filename);
336
+ writeFileSync(savedPath, buffer);
337
+ }
338
+
339
+ snapshots.push({ screen: capture.screen, path: savedPath, data: buffer.toString("base64") });
340
+ }
341
+
342
+ const content: ScreenshotResult["content"] = [];
343
+ for (const s of snapshots) {
344
+ content.push({ type: "image" as const, data: s.data, mimeType: "image/png" });
345
+ content.push({
346
+ type: "text" as const,
347
+ text: JSON.stringify({ screen: s.screen, path: s.path, viewport, scale, theme: themeLabel }, null, 2),
348
+ });
349
+ }
350
+ return { content };
351
+ } finally {
352
+ await page.close();
353
+ }
354
+ }
355
+
261
356
  // ── cleanup ─────────────────────────────────────────────────────────
262
357
 
263
358
  export async function shutdownAll() {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "openuispec",
3
- "version": "0.2.9",
3
+ "version": "0.2.11",
4
4
  "license": "MIT",
5
5
  "type": "module",
6
6
  "description": "A semantic UI specification format for AI-native, platform-native app development",
@@ -19,7 +19,10 @@
19
19
  "schema/",
20
20
  "spec/",
21
21
  "docs/",
22
- "examples/",
22
+ "examples/*/openuispec/**",
23
+ "examples/*/openuispec.yaml",
24
+ "examples/*/README.md",
25
+ "scripts/",
23
26
  "README.md",
24
27
  "LICENSE"
25
28
  ],
package/prepare/index.ts CHANGED
@@ -135,6 +135,11 @@ export interface PrepareResult {
135
135
  commit: string | null;
136
136
  branch: string | null;
137
137
  };
138
+ baseline_status?: {
139
+ output_exists: boolean;
140
+ snapshot_exists: boolean;
141
+ action_needed: string | null;
142
+ };
138
143
  summary: {
139
144
  changed: number;
140
145
  added: number;
@@ -1103,6 +1108,10 @@ function buildBootstrapPrepareResult(cwd: string, target: string, includeContent
1103
1108
  );
1104
1109
  }
1105
1110
 
1111
+ const outputDirExists = existsSync(outputDir);
1112
+ const snapshotPath = join(outputDir, ".openuispec-state.json");
1113
+ const snapshotFileExists = existsSync(snapshotPath);
1114
+
1106
1115
  return {
1107
1116
  mode: "bootstrap",
1108
1117
  project: projectName,
@@ -1116,6 +1125,13 @@ function buildBootstrapPrepareResult(cwd: string, target: string, includeContent
1116
1125
  commit: null,
1117
1126
  branch: null,
1118
1127
  },
1128
+ baseline_status: {
1129
+ output_exists: outputDirExists,
1130
+ snapshot_exists: snapshotFileExists,
1131
+ action_needed: outputDirExists && !snapshotFileExists
1132
+ ? `Baseline pending — when satisfied with the generated output, run: openuispec drift --snapshot --target ${target}`
1133
+ : null,
1134
+ },
1119
1135
  summary: {
1120
1136
  changed: 0,
1121
1137
  added: 0,
@@ -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
+ });