rivet_cms 0.1.0.pre

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 (58) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +28 -0
  4. data/Rakefile +8 -0
  5. data/app/assets/builds/rivet_cms.css +2 -0
  6. data/app/assets/builds/rivet_cms.js +9536 -0
  7. data/app/assets/builds/rivet_cms.js.map +7 -0
  8. data/app/assets/stylesheets/rivet_cms/application.tailwind.css +25 -0
  9. data/app/assets/stylesheets/rivet_cms/brand_colors.css +168 -0
  10. data/app/controllers/rivet_cms/api/docs_controller.rb +38 -0
  11. data/app/controllers/rivet_cms/application_controller.rb +5 -0
  12. data/app/controllers/rivet_cms/components_controller.rb +7 -0
  13. data/app/controllers/rivet_cms/content_types_controller.rb +61 -0
  14. data/app/controllers/rivet_cms/dashboard_controller.rb +10 -0
  15. data/app/controllers/rivet_cms/fields_controller.rb +109 -0
  16. data/app/helpers/rivet_cms/application_helper.rb +7 -0
  17. data/app/helpers/rivet_cms/brand_color_helper.rb +71 -0
  18. data/app/helpers/rivet_cms/flash_helper.rb +37 -0
  19. data/app/helpers/rivet_cms/sign_out_helper.rb +11 -0
  20. data/app/javascript/controllers/content_type_form_controller.js +53 -0
  21. data/app/javascript/controllers/field_layout_controller.js +709 -0
  22. data/app/javascript/rivet_cms.js +29 -0
  23. data/app/jobs/rivet_cms/application_job.rb +4 -0
  24. data/app/mailers/rivet_cms/application_mailer.rb +6 -0
  25. data/app/models/rivet_cms/application_record.rb +5 -0
  26. data/app/models/rivet_cms/component.rb +4 -0
  27. data/app/models/rivet_cms/content.rb +4 -0
  28. data/app/models/rivet_cms/content_type.rb +40 -0
  29. data/app/models/rivet_cms/content_value.rb +4 -0
  30. data/app/models/rivet_cms/field.rb +82 -0
  31. data/app/models/rivet_cms/field_values/base.rb +11 -0
  32. data/app/models/rivet_cms/field_values/boolean.rb +4 -0
  33. data/app/models/rivet_cms/field_values/integer.rb +4 -0
  34. data/app/models/rivet_cms/field_values/string.rb +4 -0
  35. data/app/models/rivet_cms/field_values/text.rb +4 -0
  36. data/app/services/rivet_cms/open_api_generator.rb +245 -0
  37. data/app/views/layouts/rivet_cms/application.html.erb +49 -0
  38. data/app/views/rivet_cms/api/docs/show.html.erb +47 -0
  39. data/app/views/rivet_cms/content_types/_form.html.erb +98 -0
  40. data/app/views/rivet_cms/content_types/edit.html.erb +27 -0
  41. data/app/views/rivet_cms/content_types/index.html.erb +151 -0
  42. data/app/views/rivet_cms/content_types/new.html.erb +19 -0
  43. data/app/views/rivet_cms/content_types/show.html.erb +147 -0
  44. data/app/views/rivet_cms/dashboard/index.html.erb +263 -0
  45. data/app/views/rivet_cms/fields/_form.html.erb +111 -0
  46. data/app/views/rivet_cms/fields/edit.html.erb +25 -0
  47. data/app/views/rivet_cms/fields/index.html.erb +126 -0
  48. data/app/views/rivet_cms/fields/new.html.erb +25 -0
  49. data/app/views/rivet_cms/shared/_navigation.html.erb +153 -0
  50. data/config/i18n-tasks.yml +178 -0
  51. data/config/locales/en.yml +14 -0
  52. data/config/routes.rb +56 -0
  53. data/db/migrate/20250317194359_create_core_tables.rb +90 -0
  54. data/lib/rivet_cms/engine.rb +55 -0
  55. data/lib/rivet_cms/version.rb +3 -0
  56. data/lib/rivet_cms.rb +44 -0
  57. data/lib/tasks/rivet_cms_tasks.rake +4 -0
  58. metadata +231 -0
@@ -0,0 +1,27 @@
1
+ <div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
2
+ <div class="mb-8">
3
+ <div class="flex items-center justify-between">
4
+ <div>
5
+ <h1 class="text-2xl font-bold text-gray-900">Edit Content Type</h1>
6
+ <p class="mt-1 text-sm text-gray-500">Update the details for "<%= @content_type.name %>"</p>
7
+ </div>
8
+ <div class="flex space-x-3">
9
+ <%= link_to content_type_path(@content_type), class: "inline-flex items-center px-4 py-2 border border-gray-300 rounded-md shadow-sm text-sm font-medium text-gray-700 bg-white hover:bg-gray-50 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
10
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 mr-2 -ml-1 text-gray-500" viewBox="0 0 20 20" fill="currentColor">
11
+ <path fill-rule="evenodd" d="M6.672 1.911a1 1 0 10-1.932.518l.259.966a1 1 0 001.932-.518l-.26-.966zM2.429 4.74a1 1 0 10-.517 1.932l.966.259a1 1 0 00.517-1.932l-.966-.26zm8.814-.569a1 1 0 00-1.415-1.414l-.707.707a1 1 0 101.415 1.415l.707-.708zm-7.071 7.072l.707-.707A1 1 0 003.465 9.12l-.708.707a1 1 0 001.415 1.415zm3.2-5.171a1 1 0 00-1.3 1.3l4 10a1 1 0 001.823.075l1.38-2.759 3.018 3.02a1 1 0 001.414-1.415l-3.019-3.02 2.76-1.379a1 1 0 00-.076-1.822l-10-4z" clip-rule="evenodd" />
12
+ </svg>
13
+ View
14
+ <% end %>
15
+
16
+ <%= link_to content_types_path, class: "inline-flex items-center px-4 py-2 border border-gray-300 rounded-md shadow-sm text-sm font-medium text-gray-700 bg-white hover:bg-gray-50 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
17
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 mr-2 -ml-1 text-gray-500" viewBox="0 0 20 20" fill="currentColor">
18
+ <path fill-rule="evenodd" d="M9.707 16.707a1 1 0 01-1.414 0l-6-6a1 1 0 010-1.414l6-6a1 1 0 011.414 1.414L5.414 9H17a1 1 0 110 2H5.414l4.293 4.293a1 1 0 010 1.414z" clip-rule="evenodd" />
19
+ </svg>
20
+ Back
21
+ <% end %>
22
+ </div>
23
+ </div>
24
+ </div>
25
+
26
+ <%= render "form", content_type: @content_type %>
27
+ </div>
@@ -0,0 +1,151 @@
1
+ <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
2
+ <div class="sm:flex sm:items-center mb-8">
3
+ <div class="sm:flex-auto">
4
+ <h1 class="text-2xl font-bold text-gray-900">Content Types</h1>
5
+ <p class="mt-2 text-sm text-gray-500">
6
+ A list of all the content types in your headless CMS.
7
+ </p>
8
+ </div>
9
+ <div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
10
+ <%= link_to new_content_type_path, class: "inline-flex items-center rounded-md border border-transparent bg-brand-600 px-4 py-2 text-sm font-medium text-white shadow-sm hover:bg-brand-700 focus:outline-none focus:ring-2 focus:ring-brand-500 focus:ring-offset-2 transition-all duration-200" do %>
11
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-plus -ml-1 mr-2 h-5 w-5">
12
+ <path d="M5 12h14" />
13
+ <path d="M12 5v14" />
14
+ </svg>
15
+ New Content Type
16
+ <% end %>
17
+ </div>
18
+ </div>
19
+
20
+ <div class="bg-white shadow overflow-hidden sm:rounded-lg border border-gray-100">
21
+ <% if @content_types.any? %>
22
+ <div class="overflow-x-auto">
23
+ <table class="min-w-full divide-y divide-gray-200">
24
+ <thead class="bg-gray-50">
25
+ <tr>
26
+ <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Name</th>
27
+ <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Slug</th>
28
+ <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Type</th>
29
+ <th scope="col" class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Items</th>
30
+ <th scope="col" class="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">Actions</th>
31
+ </tr>
32
+ </thead>
33
+ <tbody class="bg-white divide-y divide-gray-200">
34
+ <% @content_types.each do |content_type| %>
35
+ <tr class="hover:bg-gray-50 transition-all duration-150">
36
+ <td class="px-6 py-4 whitespace-nowrap">
37
+ <div class="flex items-center">
38
+ <div class="flex-shrink-0 h-10 w-10 flex items-center justify-center rounded-md bg-brand-50 text-brand-600 border border-brand-100">
39
+ <% if content_type.is_single %>
40
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text h-6 w-6">
41
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
42
+ <polyline points="14 2 14 8 20 8" />
43
+ <line x1="16" x2="8" y1="13" y2="13" />
44
+ <line x1="16" x2="8" y1="17" y2="17" />
45
+ <line x1="10" x2="8" y1="9" y2="9" />
46
+ </svg>
47
+ <% else %>
48
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers h-6 w-6">
49
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
50
+ <polyline points="2 17 12 22 22 17" />
51
+ <polyline points="2 12 12 17 22 12" />
52
+ </svg>
53
+ <% end %>
54
+ </div>
55
+ <div class="ml-4">
56
+ <div class="text-sm font-medium text-gray-900">
57
+ <%= link_to content_type.name, content_type_path(content_type), class: "hover:text-brand-600 transition-colors duration-150 border-b border-transparent hover:border-brand-600" %>
58
+ </div>
59
+ <% if content_type.description.present? %>
60
+ <div class="text-sm text-gray-500 truncate max-w-xs">
61
+ <%= content_type.description %>
62
+ </div>
63
+ <% end %>
64
+ </div>
65
+ </div>
66
+ </td>
67
+ <td class="px-6 py-4 whitespace-nowrap">
68
+ <div class="text-sm font-mono text-gray-900 bg-gray-50 px-2 py-1 rounded-sm border border-gray-100 inline-block"><%= content_type.slug %></div>
69
+ </td>
70
+ <td class="px-6 py-4 whitespace-nowrap">
71
+ <% if content_type.is_single %>
72
+ <span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-amber-100 text-amber-800 border border-amber-200">
73
+ Single Type
74
+ </span>
75
+ <% else %>
76
+ <span class="px-2 inline-flex text-xs leading-5 font-semibold rounded-full bg-teal-100 text-teal-800 border border-teal-200">
77
+ Collection
78
+ </span>
79
+ <% end %>
80
+ </td>
81
+ <td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
82
+ <span class="bg-gray-100 text-gray-700 px-2 py-1 rounded-full text-xs font-medium"><%= content_type.contents.count %></span>
83
+ </td>
84
+ <td class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
85
+ <div class="flex justify-end space-x-2">
86
+ <%= link_to content_type_path(content_type), class: "text-brand-500 hover:text-brand-700 transition-colors duration-150 p-1 hover:bg-brand-100 rounded-full", title: "View" do %>
87
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-eye h-5 w-5">
88
+ <path d="M2 12s3-7 10-7 10 7 10 7-3 7-10 7-10-7-10-7Z" />
89
+ <circle cx="12" cy="12" r="3" />
90
+ </svg>
91
+ <% end %>
92
+
93
+ <%= link_to content_type_fields_path(content_type), class: "text-gray-500 hover:text-gray-700 transition-colors duration-150 p-1 hover:bg-gray-200 rounded-full", title: "Structure" do %>
94
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layout-grid h-5 w-5">
95
+ <path d="M12 3v18"/><rect width="18" height="18" x="3" y="3" rx="2"/><path d="M3 9h18"/><path d="M3 15h18"/>
96
+ </svg>
97
+ <% end %>
98
+
99
+ <%= link_to edit_content_type_path(content_type), class: "text-amber-500 hover:text-amber-700 transition-colors duration-150 p-1 hover:bg-amber-100 rounded-full", title: "Edit" do %>
100
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pencil h-5 w-5">
101
+ <path d="M17 3a2.85 2.85 0 1 1 4 4L7.5 20.5 2 22l1.5-5.5Z" />
102
+ <path d="m15 5 4 4" />
103
+ </svg>
104
+ <% end %>
105
+
106
+ <%= button_to content_type_path(content_type), method: :delete, class: "text-red-500 hover:text-red-700 transition-colors duration-150 p-1 hover:bg-red-100 rounded-full", title: "Delete", form: { data: { turbo_confirm: "Are you sure you want to delete this content type? This will also delete all contents associated with it." } } do %>
107
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-trash-2 h-5 w-5">
108
+ <path d="M3 6h18" />
109
+ <path d="M19 6v14c0 1-1 2-2 2H7c-1 0-2-1-2-2V6" />
110
+ <path d="M8 6V4c0-1 1-2 2-2h4c1 0 2 1 2 2v2" />
111
+ <line x1="10" x2="10" y1="11" y2="17" />
112
+ <line x1="14" x2="14" y1="11" y2="17" />
113
+ </svg>
114
+ <% end %>
115
+ </div>
116
+ </td>
117
+ </tr>
118
+ <% end %>
119
+ </tbody>
120
+ </table>
121
+ </div>
122
+ <% else %>
123
+ <div class="text-center py-16 px-6">
124
+ <div class="relative mx-auto h-24 w-24 text-gray-400 bg-gray-50 rounded-full border border-gray-100 flex items-center justify-center">
125
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers h-12 w-12">
126
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
127
+ <polyline points="2 17 12 22 22 17" />
128
+ <polyline points="2 12 12 17 22 12" />
129
+ </svg>
130
+ <div class="absolute -top-1 -right-1 h-6 w-6 bg-brand-100 rounded-full border border-brand-200 flex items-center justify-center">
131
+ <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-plus h-4 w-4 text-brand-600">
132
+ <path d="M5 12h14" />
133
+ <path d="M12 5v14" />
134
+ </svg>
135
+ </div>
136
+ </div>
137
+ <h3 class="mt-4 text-sm font-medium text-gray-900">No content types</h3>
138
+ <p class="mt-1 text-sm text-gray-500">Get started by creating a new content type.</p>
139
+ <div class="mt-6">
140
+ <%= link_to new_content_type_path, class: "inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-brand-600 hover:bg-brand-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
141
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-plus -ml-1 mr-2 h-5 w-5">
142
+ <path d="M5 12h14" />
143
+ <path d="M12 5v14" />
144
+ </svg>
145
+ New Content Type
146
+ <% end %>
147
+ </div>
148
+ </div>
149
+ <% end %>
150
+ </div>
151
+ </div>
@@ -0,0 +1,19 @@
1
+ <div class="max-w-4xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
2
+ <div class="mb-8">
3
+ <div class="flex items-center justify-between">
4
+ <div>
5
+ <h1 class="text-2xl font-bold text-gray-900">New Content Type</h1>
6
+ </div>
7
+ <div>
8
+ <%= link_to content_types_path, class: "inline-flex items-center px-4 py-2 border border-gray-300 rounded-md shadow-sm text-sm font-medium text-gray-700 bg-white hover:bg-gray-50 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
9
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 mr-2 -ml-1 text-gray-500" viewBox="0 0 20 20" fill="currentColor">
10
+ <path fill-rule="evenodd" d="M9.707 16.707a1 1 0 01-1.414 0l-6-6a1 1 0 010-1.414l6-6a1 1 0 011.414 1.414L5.414 9H17a1 1 0 110 2H5.414l4.293 4.293a1 1 0 010 1.414z" clip-rule="evenodd" />
11
+ </svg>
12
+ Back to Content Types
13
+ <% end %>
14
+ </div>
15
+ </div>
16
+ </div>
17
+
18
+ <%= render "form", content_type: @content_type %>
19
+ </div>
@@ -0,0 +1,147 @@
1
+ <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8 space-y-8">
2
+ <div class="sm:flex sm:items-center">
3
+ <div class="sm:flex-auto">
4
+ <div class="flex items-center">
5
+ <h1 class="text-2xl font-bold text-gray-900"><%= @content_type.name %></h1>
6
+ <span class="ml-3 inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium <%= @content_type.is_single ? 'bg-amber-100 text-amber-800 border border-amber-200' : 'bg-teal-100 text-teal-800 border border-teal-200' %>">
7
+ <%= @content_type.is_single ? "Single Type" : "Collection" %>
8
+ </span>
9
+ </div>
10
+ <p class="mt-2 text-sm text-gray-700">
11
+ API Endpoint: <code class="bg-gray-50 px-3 py-1 rounded font-mono text-sm border border-gray-100"><%= @content_type.is_single ? "/api/v1/#{@content_type.slug}" : "/api/v1/#{@content_type.slug}" %></code>
12
+ </p>
13
+ <% if @content_type.description.present? %>
14
+ <p class="mt-2 text-sm text-gray-500"><%= @content_type.description %></p>
15
+ <% end %>
16
+ </div>
17
+ <div class="mt-4 sm:mt-0 sm:ml-16 sm:flex-none">
18
+ <div class="flex space-x-3">
19
+ <%= link_to content_types_path, class: "inline-flex items-center rounded-md border border-gray-300 bg-white px-4 py-2 text-sm font-medium text-gray-700 shadow-sm hover:bg-gray-50 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-brand-500 focus:ring-offset-2 transition-all duration-200" do %>
20
+ <svg class="-ml-1 mr-2 h-5 w-5 text-gray-500" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
21
+ <path fill-rule="evenodd" d="M9.707 16.707a1 1 0 01-1.414 0l-6-6a1 1 0 010-1.414l6-6a1 1 0 011.414 1.414L5.414 9H17a1 1 0 110 2H5.414l4.293 4.293a1 1 0 010 1.414z" clip-rule="evenodd" />
22
+ </svg>
23
+ Back
24
+ <% end %>
25
+ <%= link_to edit_content_type_path(@content_type), class: "inline-flex items-center rounded-md border border-gray-300 bg-white px-4 py-2 text-sm font-medium text-gray-700 shadow-sm hover:bg-gray-50 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-brand-500 focus:ring-offset-2 transition-all duration-200" do %>
26
+ <svg class="-ml-1 mr-2 h-5 w-5 text-gray-500" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
27
+ <path d="M13.586 3.586a2 2 0 112.828 2.828l-.793.793-2.828-2.828.793-.793zM11.379 5.793L3 14.172V17h2.828l8.38-8.379-2.83-2.828z" />
28
+ </svg>
29
+ Edit Content Type
30
+ <% end %>
31
+ <% if @content_type.fields.any? %>
32
+ <%= link_to "#", class: "inline-flex items-center rounded-md border border-transparent bg-brand-600 px-4 py-2 text-sm font-medium text-white shadow-sm hover:bg-brand-700 focus:outline-none focus:ring-2 focus:ring-brand-500 focus:ring-offset-2 transition-all duration-200" do %>
33
+ <svg class="-ml-1 mr-2 h-5 w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
34
+ <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd" />
35
+ </svg>
36
+ New <%= @content_type.name.singularize %>
37
+ <% end %>
38
+ <% end %>
39
+ </div>
40
+ </div>
41
+ </div>
42
+
43
+ <!-- Content Items -->
44
+ <div class="bg-white shadow-md rounded-lg overflow-hidden border border-gray-100">
45
+ <div class="px-6 py-5 border-b border-gray-200 bg-gray-50">
46
+ <div class="flex justify-between items-center">
47
+ <h2 class="text-lg font-medium text-gray-900">
48
+ <%= @content_type.is_single ? @content_type.name : "#{@content_type.name} Items" %>
49
+ </h2>
50
+ <span class="bg-gray-100 text-gray-700 px-2 py-1 rounded-full text-xs font-medium border border-gray-200"><%= pluralize(@contents.count, 'item') %></span>
51
+ </div>
52
+ </div>
53
+
54
+ <% if @contents.any? %>
55
+ <div class="overflow-x-auto">
56
+ <table class="min-w-full divide-y divide-gray-200">
57
+ <thead class="bg-gray-50">
58
+ <tr>
59
+ <th scope="col" class="py-3.5 pl-6 pr-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Title</th>
60
+ <th scope="col" class="px-3 py-3.5 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Status</th>
61
+ <th scope="col" class="px-3 py-3.5 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Created</th>
62
+ <th scope="col" class="px-3 py-3.5 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">Updated</th>
63
+ <th scope="col" class="relative py-3.5 pl-3 pr-6 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">
64
+ Actions
65
+ </th>
66
+ </tr>
67
+ </thead>
68
+ <tbody class="divide-y divide-gray-200 bg-white">
69
+ <% @contents.each do |content| %>
70
+ <tr class="hover:bg-gray-50 transition-all duration-150">
71
+ <td class="whitespace-nowrap py-4 pl-6 pr-3 text-sm font-medium text-gray-900"><%= content.title %></td>
72
+ <td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500">
73
+ <% if content.status == 'published' %>
74
+ <span class="inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium bg-brand-100 text-brand-800 border border-brand-200">
75
+ Published
76
+ </span>
77
+ <% elsif content.status == 'draft' %>
78
+ <span class="inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium bg-amber-100 text-amber-800 border border-amber-200">
79
+ Draft
80
+ </span>
81
+ <% else %>
82
+ <span class="inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium bg-gray-100 text-gray-800 border border-gray-200">
83
+ <%= content.status.capitalize %>
84
+ </span>
85
+ <% end %>
86
+ </td>
87
+ <td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500"><%= content.created_at.strftime("%b %d, %Y") %></td>
88
+ <td class="whitespace-nowrap px-3 py-4 text-sm text-gray-500"><%= content.updated_at.strftime("%b %d, %Y") %></td>
89
+ <td class="relative whitespace-nowrap py-4 pl-3 pr-6 text-right text-sm font-medium">
90
+ <div class="flex justify-end space-x-2">
91
+ <%= link_to "#", class: "text-brand-600 hover:text-brand-900 transition-colors duration-150 p-1 hover:bg-brand-50 rounded-full" do %>
92
+ <span class="sr-only">Edit</span>
93
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
94
+ <path d="M13.586 3.586a2 2 0 112.828 2.828l-.793.793-2.828-2.828.793-.793zM11.379 5.793L3 14.172V17h2.828l8.38-8.379-2.83-2.828z" />
95
+ </svg>
96
+ <% end %>
97
+ <%= button_to "#", method: :delete, class: "text-red-600 hover:text-red-900 transition-colors duration-150 p-1 hover:bg-red-50 rounded-full", form: { data: { turbo_confirm: "Are you sure? This will permanently delete this content." } } do %>
98
+ <span class="sr-only">Delete</span>
99
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
100
+ <path fill-rule="evenodd" d="M9 2a1 1 0 00-.894.553L7.382 4H4a1 1 0 000 2v10a2 2 0 002 2h8a2 2 0 002-2V6a1 1 0 100-2h-3.382l-.724-1.447A1 1 0 0011 2H9zM7 8a1 1 0 012 0v6a1 1 0 11-2 0V8zm5-1a1 1 0 00-1 1v6a1 1 0 102 0V8a1 1 0 00-1-1z" clip-rule="evenodd" />
101
+ </svg>
102
+ <% end %>
103
+ </div>
104
+ </td>
105
+ </tr>
106
+ <% end %>
107
+ </tbody>
108
+ </table>
109
+ </div>
110
+ <% else %>
111
+ <div class="text-center py-16 px-6">
112
+ <div class="relative mx-auto h-24 w-24 text-gray-400 bg-gray-50 rounded-full border border-gray-100 flex items-center justify-center">
113
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-12 w-12" fill="none" viewBox="0 0 24 24" stroke="currentColor">
114
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5" d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
115
+ </svg>
116
+ <div class="absolute -top-1 -right-1 h-6 w-6 bg-brand-100 rounded-full border border-brand-200 flex items-center justify-center">
117
+ <svg xmlns="http://www.w3.org/2000/svg" class="h-4 w-4 text-brand-600" viewBox="0 0 20 20" fill="currentColor">
118
+ <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd" />
119
+ </svg>
120
+ </div>
121
+ </div>
122
+ <h3 class="mt-4 text-sm font-medium text-gray-900">No content</h3>
123
+ <% if @content_type.fields.any? %>
124
+ <p class="mt-1 text-sm text-gray-500">Get started by creating content for this content type.</p>
125
+ <div class="mt-6">
126
+ <%= link_to "#", class: "inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-brand-600 hover:bg-brand-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
127
+ <svg class="-ml-1 mr-2 h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
128
+ <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd" />
129
+ </svg>
130
+ New <%= @content_type.name.singularize %>
131
+ <% end %>
132
+ </div>
133
+ <% else %>
134
+ <p class="mt-1 text-sm text-gray-500">You must add at least one field to this content type before you can create any content.</p>
135
+ <div class="mt-6">
136
+ <%= link_to "#", class: "inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-brand-600 hover:bg-brand-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-brand-500 transition-all duration-200" do %>
137
+ <svg class="-ml-1 mr-2 h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
138
+ <path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd" />
139
+ </svg>
140
+ Add Field
141
+ <% end %>
142
+ </div>
143
+ <% end %>
144
+ </div>
145
+ <% end %>
146
+ </div>
147
+ </div>
@@ -0,0 +1,263 @@
1
+ <div>
2
+ <div class="bg-white rounded-lg shadow-sm border border-gray-200 mb-8 p-6">
3
+ <h1 class="text-2xl font-semibold text-gray-900">Welcome to RivetCMS</h1>
4
+ <p class="mt-2 text-gray-500">Your headless CMS for structured content</p>
5
+
6
+ <div class="grid grid-cols-1 md:grid-cols-3 gap-6 mt-6">
7
+ <div class="bg-white rounded-lg p-4 border border-gray-200 shadow-sm">
8
+ <div class="flex items-center">
9
+ <div class="p-2 rounded-md bg-brand-50 text-brand-600 border border-brand-100">
10
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers">
11
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
12
+ <polyline points="2 17 12 22 22 17" />
13
+ <polyline points="2 12 12 17 22 12" />
14
+ </svg>
15
+ </div>
16
+ <div class="ml-4">
17
+ <p class="text-sm text-gray-500">Content Types</p>
18
+ <p class="text-xl font-medium text-gray-900"><%= @content_types.count %></p>
19
+ </div>
20
+ </div>
21
+ </div>
22
+
23
+ <div class="bg-white rounded-lg p-4 border border-gray-200 shadow-sm">
24
+ <div class="flex items-center">
25
+ <div class="p-2 rounded-md bg-brand-50 text-brand-600 border border-brand-100">
26
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text">
27
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
28
+ <polyline points="14 2 14 8 20 8" />
29
+ <line x1="16" x2="8" y1="13" y2="13" />
30
+ <line x1="16" x2="8" y1="17" y2="17" />
31
+ <line x1="10" x2="8" y1="9" y2="9" />
32
+ </svg>
33
+ </div>
34
+ <div class="ml-4">
35
+ <p class="text-sm text-gray-500">Content Items</p>
36
+ <p class="text-xl font-medium text-gray-900"><%= @recent_contents.count %></p>
37
+ </div>
38
+ </div>
39
+ </div>
40
+
41
+ <div class="bg-white rounded-lg p-4 border border-gray-200 shadow-sm">
42
+ <div class="flex items-center">
43
+ <div class="p-2 rounded-md bg-brand-50 text-brand-600 border border-brand-100">
44
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layout-grid">
45
+ <rect width="7" height="7" x="3" y="3" rx="1" />
46
+ <rect width="7" height="7" x="14" y="3" rx="1" />
47
+ <rect width="7" height="7" x="14" y="14" rx="1" />
48
+ <rect width="7" height="7" x="3" y="14" rx="1" />
49
+ </svg>
50
+ </div>
51
+ <div class="ml-4">
52
+ <p class="text-sm text-gray-500">Components</p>
53
+ <p class="text-xl font-medium text-gray-900"><%= @components.count %></p>
54
+ </div>
55
+ </div>
56
+ </div>
57
+ </div>
58
+ </div>
59
+
60
+ <div class="mt-6 grid grid-cols-1 gap-6 lg:grid-cols-2">
61
+ <!-- Content Types Card -->
62
+ <div class="bg-white overflow-hidden shadow-sm rounded-lg border border-gray-200">
63
+ <div class="px-4 py-5 sm:px-6 flex justify-between items-center">
64
+ <h2 class="text-lg font-medium text-gray-900 flex items-center">
65
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers mr-2 text-brand-600">
66
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
67
+ <polyline points="2 17 12 22 22 17" />
68
+ <polyline points="2 12 12 17 22 12" />
69
+ </svg>
70
+ Content Types
71
+ </h2>
72
+ <%= link_to new_content_type_path, class: "inline-flex items-center rounded-md bg-brand-600 px-3 py-2 text-sm font-semibold text-white shadow-sm hover:bg-brand-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-brand-600" do %>
73
+ <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-plus mr-1">
74
+ <path d="M5 12h14" />
75
+ <path d="M12 5v14" />
76
+ </svg>
77
+ New
78
+ <% end %>
79
+ </div>
80
+ <div class="border-t border-gray-200 px-4 py-5 sm:p-6">
81
+ <% if @content_types.any? %>
82
+ <div class="flow-root">
83
+ <ul role="list" class="-my-5 divide-y divide-gray-200">
84
+ <% @content_types.each do |content_type| %>
85
+ <li class="py-4 hover:bg-gray-50 rounded-md px-2">
86
+ <div class="flex items-center space-x-4">
87
+ <div class="flex-shrink-0">
88
+ <span class="inline-flex h-10 w-10 items-center justify-center rounded-md bg-brand-50 text-brand-600 border border-brand-100">
89
+ <% if content_type.is_single %>
90
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text">
91
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
92
+ <polyline points="14 2 14 8 20 8" />
93
+ <line x1="16" x2="8" y1="13" y2="13" />
94
+ <line x1="16" x2="8" y1="17" y2="17" />
95
+ <line x1="10" x2="8" y1="9" y2="9" />
96
+ </svg>
97
+ <% else %>
98
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers">
99
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
100
+ <polyline points="2 17 12 22 22 17" />
101
+ <polyline points="2 12 12 17 22 12" />
102
+ </svg>
103
+ <% end %>
104
+ </span>
105
+ </div>
106
+ <div class="min-w-0 flex-1">
107
+ <p class="truncate text-sm font-medium text-gray-900"><%= content_type.name %></p>
108
+ <p class="truncate text-sm text-gray-500"><%= content_type.is_single ? "Single Type" : "Collection" %> • <%= content_type.contents.count %> items</p>
109
+ </div>
110
+ <div>
111
+ <%= link_to content_type_path(content_type), class: "inline-flex items-center rounded-md bg-white px-2.5 py-1.5 text-sm font-semibold text-gray-900 shadow-sm ring-1 ring-inset ring-gray-300 hover:bg-gray-50" do %>
112
+ <% if content_type.is_single %>
113
+ Edit
114
+ <% else %>
115
+ View All
116
+ <% end %>
117
+ <% end %>
118
+ </div>
119
+ </div>
120
+ </li>
121
+ <% end %>
122
+ </ul>
123
+ </div>
124
+ <% else %>
125
+ <div class="text-center py-6">
126
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-folder-plus mx-auto h-12 w-12 text-gray-400">
127
+ <path d="M4 20h16a2 2 0 0 0 2-2V8a2 2 0 0 0-2-2h-7.93a2 2 0 0 1-1.66-.9l-.82-1.2A2 2 0 0 0 7.93 3H4a2 2 0 0 0-2 2v13c0 1.1.9 2 2 2Z" />
128
+ <line x1="12" x2="12" y1="10" y2="16" />
129
+ <line x1="9" x2="15" y1="13" y2="13" />
130
+ </svg>
131
+ <h3 class="mt-2 text-sm font-semibold text-gray-900">No content types</h3>
132
+ <p class="mt-1 text-sm text-gray-500">Get started by creating a new content type.</p>
133
+ <div class="mt-6">
134
+ <%= link_to new_content_type_path, class: "inline-flex items-center rounded-md bg-brand-600 px-3 py-2 text-sm font-semibold text-white shadow-sm hover:bg-brand-500 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-brand-600" do %>
135
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-plus mr-2">
136
+ <path d="M5 12h14" />
137
+ <path d="M12 5v14" />
138
+ </svg>
139
+ New Content Type
140
+ <% end %>
141
+ </div>
142
+ </div>
143
+ <% end %>
144
+ </div>
145
+ </div>
146
+
147
+ <!-- Recent Content Card -->
148
+ <div class="bg-white overflow-hidden shadow-sm rounded-lg border border-gray-200">
149
+ <div class="px-4 py-5 sm:px-6 flex justify-between items-center">
150
+ <h2 class="text-lg font-medium text-gray-900 flex items-center">
151
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text mr-2 text-brand-600">
152
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
153
+ <polyline points="14 2 14 8 20 8" />
154
+ <line x1="16" x2="8" y1="13" y2="13" />
155
+ <line x1="16" x2="8" y1="17" y2="17" />
156
+ <line x1="10" x2="8" y1="9" y2="9" />
157
+ </svg>
158
+ Recent Content
159
+ </h2>
160
+ </div>
161
+ <div class="border-t border-gray-200 px-4 py-5 sm:p-6">
162
+ <% if @recent_contents.any? %>
163
+ <div class="flow-root">
164
+ <ul role="list" class="-my-5 divide-y divide-gray-200">
165
+ <% @recent_contents.each do |content| %>
166
+ <li class="py-4 hover:bg-gray-50 rounded-md px-2">
167
+ <div class="flex items-center space-x-4">
168
+ <div class="flex-shrink-0">
169
+ <span class="inline-flex h-10 w-10 items-center justify-center rounded-md bg-brand-50 text-brand-600 border border-brand-100">
170
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file">
171
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
172
+ <polyline points="14 2 14 8 20 8" />
173
+ </svg>
174
+ </span>
175
+ </div>
176
+ <div class="min-w-0 flex-1">
177
+ <p class="truncate text-sm font-medium text-gray-900"><%= content.title %></p>
178
+ <p class="truncate text-sm text-gray-500">
179
+ <%= content.content_type.name %> •
180
+ <span class="inline-flex items-center rounded-full px-2 py-0.5 text-xs font-medium <%= content.status == 'published' ? 'bg-green-100 text-green-800 border border-green-200' : 'bg-yellow-100 text-yellow-800 border border-yellow-200' %>">
181
+ <% if content.status == 'published' %>
182
+ <svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-check mr-1"><polyline points="20 6 9 17 4 12"/></svg>
183
+ <% else %>
184
+ <svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-pencil mr-1"><path d="M17 3a2.85 2.85 0 1 1 4 4L7.5 20.5 2 22l1.5-5.5Z"/><path d="m15 5 4 4"/></svg>
185
+ <% end %>
186
+ <%= content.status.capitalize %>
187
+ </span>
188
+ </p>
189
+ </div>
190
+ <div>
191
+ <%= link_to "Edit", content_type_path(content.content_type), class: "inline-flex items-center rounded-md bg-white px-2.5 py-1.5 text-sm font-semibold text-gray-900 shadow-sm ring-1 ring-inset ring-gray-300 hover:bg-gray-50" %>
192
+ </div>
193
+ </div>
194
+ </li>
195
+ <% end %>
196
+ </ul>
197
+ </div>
198
+ <% else %>
199
+ <div class="text-center py-6">
200
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text mx-auto h-12 w-12 text-gray-400">
201
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
202
+ <polyline points="14 2 14 8 20 8" />
203
+ <line x1="16" x2="8" y1="13" y2="13" />
204
+ <line x1="16" x2="8" y1="17" y2="17" />
205
+ <line x1="10" x2="8" y1="9" y2="9" />
206
+ </svg>
207
+ <h3 class="mt-2 text-sm font-semibold text-gray-900">No content</h3>
208
+ <p class="mt-1 text-sm text-gray-500">Start creating content for your content types.</p>
209
+ </div>
210
+ <% end %>
211
+ </div>
212
+ </div>
213
+ </div>
214
+
215
+ <!-- API Information Card -->
216
+ <div class="mt-6 bg-white overflow-hidden shadow-sm rounded-lg border border-gray-200">
217
+ <div class="px-4 py-5 sm:px-6 flex items-center">
218
+ <svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-code mr-2 text-brand-600">
219
+ <polyline points="16 18 22 12 16 6" />
220
+ <polyline points="8 6 2 12 8 18" />
221
+ </svg>
222
+ <h2 class="text-lg font-medium text-gray-900">API Endpoints</h2>
223
+ </div>
224
+ <div class="border-t border-gray-200 px-4 py-5 sm:p-6">
225
+ <div class="prose pbrand-sm max-w-none">
226
+ <div class="flex items-center mb-4">
227
+ <p class="text-gray-700">Your content is accessible through the following API endpoints:</p>
228
+ <%= link_to api_docs_path, class: "ml-4 inline-flex items-center text-sm text-brand-600 hover:text-brand-800 font-medium", target: "_blank" do %>
229
+ <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-external-link mr-1"><path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6"/><polyline points="15 3 21 3 21 9"/><line x1="10" x2="21" y1="14" y2="3"/></svg>
230
+ View API Docs
231
+ <% end %>
232
+ </div>
233
+
234
+ <h3 class="font-medium text-gray-900 py-2 flex items-center">
235
+ <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-layers mr-2 text-brand-600">
236
+ <polygon points="12 2 2 7 12 12 22 7 12 2" />
237
+ <polyline points="2 17 12 22 22 17" />
238
+ <polyline points="2 12 12 17 22 12" />
239
+ </svg>
240
+ Collection Types
241
+ </h3>
242
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">GET /api/v1/:slug</code> <span class="text-gray-500">- List all items</span></pre>
243
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">GET /api/v1/:slug/:id</code> <span class="text-gray-500">- Get a specific item</span></pre>
244
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">POST /api/v1/:slug</code> <span class="text-gray-500">- Create a new item</span></pre>
245
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">PUT/PATCH /api/v1/:slug/:id</code> <span class="text-gray-500">- Update an item</span></pre>
246
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">DELETE /api/v1/:slug/:id</code> <span class="text-gray-500">- Delete an item</span></pre>
247
+
248
+ <h3 class="font-medium text-gray-900 py-2 flex items-center mt-4">
249
+ <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-file-text mr-2 text-brand-600">
250
+ <path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z" />
251
+ <polyline points="14 2 14 8 20 8" />
252
+ <line x1="16" x2="8" y1="13" y2="13" />
253
+ <line x1="16" x2="8" y1="17" y2="17" />
254
+ <line x1="10" x2="8" y1="9" y2="9" />
255
+ </svg>
256
+ Single Types
257
+ </h3>
258
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">GET /api/v1/:slug</code> <span class="text-gray-500">- Get the single item</span></pre>
259
+ <pre class="bg-gray-50 p-3 rounded-md text-sm border border-gray-200 overflow-x-auto"><code class="text-gray-800">PUT/PATCH/POST /api/v1/:slug</code> <span class="text-gray-500">- Update the single item</span></pre>
260
+ </div>
261
+ </div>
262
+ </div>
263
+ </div>