focomy 0.1.105__py3-none-any.whl → 0.1.106__py3-none-any.whl

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.
@@ -28,16 +28,10 @@ fields:
28
28
  - name: fields_config
29
29
  type: json
30
30
  label: フィールド設定
31
- hint: |
32
- フォームフィールドをJSON配列で定義します。
33
- 例: [{"name": "email", "type": "email", "label": "メールアドレス", "required": true}]
34
31
 
35
32
  - name: steps
36
33
  type: json
37
- label: ステップ設定
38
- hint: |
39
- ウィザード形式の場合、ステップを定義します。
40
- 例: [{"title": "基本情報", "fields": ["name", "email"]}, {"title": "詳細", "fields": ["message"]}]
34
+ label: ステップ設定(ウィザード形式)
41
35
 
42
36
  - name: submit_label
43
37
  type: string
core/main.py CHANGED
@@ -81,7 +81,7 @@ class SecurityHeadersMiddleware(BaseHTTPMiddleware):
81
81
  # CSP for public pages (more restrictive)
82
82
  PUBLIC_CSP = (
83
83
  "default-src 'self'; "
84
- "script-src 'self' 'unsafe-inline' https://www.googletagmanager.com https://www.google-analytics.com; "
84
+ "script-src 'self' 'unsafe-inline' https://www.googletagmanager.com https://www.google-analytics.com https://unpkg.com; "
85
85
  "style-src 'self' 'unsafe-inline' https://fonts.googleapis.com; "
86
86
  "img-src 'self' data: https: blob:; "
87
87
  "font-src 'self' https://fonts.gstatic.com; "
@@ -292,6 +292,56 @@
292
292
  </div>
293
293
  </div>
294
294
  </div>
295
+ {% elif field.name == 'steps' %}
296
+ {# === Visual Step Builder for steps === #}
297
+ <input type="hidden" id="{{ field.name }}" name="{{ field.name }}" value='{{ entity.get(field.name, "[]") | tojson if entity and entity.get(field.name) else "[]" }}'>
298
+
299
+ <div class="step-builder" id="step-builder">
300
+ <p class="form-hint" style="margin-bottom: 0.75rem;">
301
+ 複数ステップに分けるとウィザード形式のフォームになります。空の場合は1ページ形式です。
302
+ </p>
303
+ <div class="step-builder-toolbar">
304
+ <button type="button" class="btn btn-primary btn-sm" onclick="StepBuilder.showAddModal()">
305
+ + ステップ追加
306
+ </button>
307
+ </div>
308
+ <ul class="step-builder-list" id="step-builder-list">
309
+ {# Steps rendered by JavaScript #}
310
+ </ul>
311
+ <div class="step-builder-empty" id="step-builder-empty" style="display: none;">
312
+ <p class="text-muted">ステップがありません。1ページ形式で表示されます。</p>
313
+ </div>
314
+ </div>
315
+
316
+ {# Step Edit Modal #}
317
+ <div id="step-modal" class="modal" style="display: none;">
318
+ <div class="modal-overlay" onclick="StepBuilder.hideModal()"></div>
319
+ <div class="modal-content">
320
+ <div class="modal-header">
321
+ <h3 id="step-modal-title">ステップ追加</h3>
322
+ <button type="button" class="modal-close" onclick="StepBuilder.hideModal()">&times;</button>
323
+ </div>
324
+ <div class="modal-body">
325
+ <input type="hidden" id="step-edit-index" value="-1">
326
+
327
+ <div class="form-group">
328
+ <label class="form-label" for="step-title">ステップタイトル <span style="color: var(--error);">*</span></label>
329
+ <input type="text" id="step-title" class="form-input" placeholder="例: 基本情報">
330
+ </div>
331
+
332
+ <div class="form-group">
333
+ <label class="form-label">このステップに含めるフィールド</label>
334
+ <div id="step-fields-selector" class="step-fields-selector">
335
+ <p class="text-muted" id="step-no-fields">先にフィールドを追加してください</p>
336
+ </div>
337
+ </div>
338
+ </div>
339
+ <div class="modal-footer">
340
+ <button type="button" class="btn btn-secondary" onclick="StepBuilder.hideModal()">キャンセル</button>
341
+ <button type="button" class="btn btn-primary" onclick="StepBuilder.saveStep()">保存</button>
342
+ </div>
343
+ </div>
344
+ </div>
295
345
  {% else %}
296
346
  <textarea
297
347
  id="{{ field.name }}"
@@ -940,6 +990,94 @@
940
990
  padding: 1rem 1.5rem;
941
991
  border-top: 1px solid var(--border);
942
992
  }
993
+
994
+ /* === Step Builder Styles === */
995
+ .step-builder {
996
+ border: 1px solid var(--border);
997
+ border-radius: 0.5rem;
998
+ background: var(--bg);
999
+ }
1000
+ .step-builder-toolbar {
1001
+ padding: 0.75rem;
1002
+ border-bottom: 1px solid var(--border);
1003
+ background: var(--card);
1004
+ border-radius: 0.5rem 0.5rem 0 0;
1005
+ }
1006
+ .step-builder-list {
1007
+ list-style: none;
1008
+ padding: 0.75rem;
1009
+ margin: 0;
1010
+ min-height: 60px;
1011
+ }
1012
+ .step-builder-item {
1013
+ display: flex;
1014
+ align-items: center;
1015
+ gap: 0.75rem;
1016
+ padding: 0.75rem 1rem;
1017
+ background: var(--card);
1018
+ border: 1px solid var(--border);
1019
+ border-radius: 0.375rem;
1020
+ margin-bottom: 0.5rem;
1021
+ }
1022
+ .step-builder-item:last-child {
1023
+ margin-bottom: 0;
1024
+ }
1025
+ .step-builder-item.sortable-ghost {
1026
+ opacity: 0.4;
1027
+ }
1028
+ .step-builder-item.sortable-chosen {
1029
+ border-color: var(--primary);
1030
+ box-shadow: 0 2px 8px rgba(0,0,0,0.1);
1031
+ }
1032
+ .step-builder-drag {
1033
+ cursor: grab;
1034
+ color: var(--text-muted);
1035
+ font-size: 1rem;
1036
+ }
1037
+ .step-builder-drag:active {
1038
+ cursor: grabbing;
1039
+ }
1040
+ .step-builder-info {
1041
+ flex: 1;
1042
+ min-width: 0;
1043
+ }
1044
+ .step-builder-title {
1045
+ font-weight: 500;
1046
+ }
1047
+ .step-builder-fields {
1048
+ font-size: 0.75rem;
1049
+ color: var(--text-muted);
1050
+ margin-top: 0.25rem;
1051
+ }
1052
+ .step-builder-actions {
1053
+ display: flex;
1054
+ gap: 0.5rem;
1055
+ }
1056
+ .step-builder-empty {
1057
+ padding: 1.5rem;
1058
+ text-align: center;
1059
+ }
1060
+ .step-fields-selector {
1061
+ max-height: 200px;
1062
+ overflow-y: auto;
1063
+ border: 1px solid var(--border);
1064
+ border-radius: 0.375rem;
1065
+ padding: 0.5rem;
1066
+ }
1067
+ .step-field-option {
1068
+ display: flex;
1069
+ align-items: center;
1070
+ gap: 0.5rem;
1071
+ padding: 0.375rem 0.5rem;
1072
+ border-radius: 0.25rem;
1073
+ cursor: pointer;
1074
+ }
1075
+ .step-field-option:hover {
1076
+ background: var(--bg);
1077
+ }
1078
+ .step-field-option input {
1079
+ width: auto;
1080
+ }
943
1081
  </style>
944
1082
 
945
1083
  {% set has_blocks = namespace(value=false) %}
@@ -1456,7 +1594,215 @@ const FormBuilder = (function() {
1456
1594
  hideModal,
1457
1595
  toggleOptionsEditor,
1458
1596
  addOption,
1459
- saveField
1597
+ saveField,
1598
+ getFields: function() { return fields; }
1599
+ };
1600
+ })();
1601
+
1602
+ // === Step Builder ===
1603
+ const StepBuilder = (function() {
1604
+ let steps = [];
1605
+
1606
+ function init() {
1607
+ const input = document.getElementById('steps');
1608
+ if (!input) return;
1609
+
1610
+ // Parse existing data
1611
+ try {
1612
+ const data = JSON.parse(input.value || '[]');
1613
+ steps = Array.isArray(data) ? data : [];
1614
+ } catch (e) {
1615
+ console.error('Failed to parse steps:', e);
1616
+ steps = [];
1617
+ }
1618
+
1619
+ render();
1620
+ initSortable();
1621
+ }
1622
+
1623
+ function initSortable() {
1624
+ const list = document.getElementById('step-builder-list');
1625
+ if (!list || typeof Sortable === 'undefined') return;
1626
+
1627
+ new Sortable(list, {
1628
+ animation: 150,
1629
+ handle: '.step-builder-drag',
1630
+ ghostClass: 'sortable-ghost',
1631
+ chosenClass: 'sortable-chosen',
1632
+ onEnd: function() {
1633
+ const items = list.querySelectorAll('.step-builder-item');
1634
+ const newSteps = [];
1635
+ items.forEach(item => {
1636
+ const index = parseInt(item.dataset.index);
1637
+ if (steps[index]) {
1638
+ newSteps.push(steps[index]);
1639
+ }
1640
+ });
1641
+ steps = newSteps;
1642
+ save();
1643
+ render();
1644
+ }
1645
+ });
1646
+ }
1647
+
1648
+ function render() {
1649
+ const list = document.getElementById('step-builder-list');
1650
+ const empty = document.getElementById('step-builder-empty');
1651
+ if (!list) return;
1652
+
1653
+ if (steps.length === 0) {
1654
+ list.innerHTML = '';
1655
+ if (empty) empty.style.display = 'block';
1656
+ return;
1657
+ }
1658
+
1659
+ if (empty) empty.style.display = 'none';
1660
+
1661
+ list.innerHTML = steps.map((step, index) => {
1662
+ const fieldLabels = getFieldLabels(step.fields || []);
1663
+ return `
1664
+ <li class="step-builder-item" data-index="${index}">
1665
+ <span class="step-builder-drag">&#9776;</span>
1666
+ <div class="step-builder-info">
1667
+ <div class="step-builder-title">ステップ${index + 1}: ${escapeHtml(step.title)}</div>
1668
+ <div class="step-builder-fields">フィールド: ${fieldLabels || '(なし)'}</div>
1669
+ </div>
1670
+ <div class="step-builder-actions">
1671
+ <button type="button" class="btn btn-secondary btn-sm" onclick="StepBuilder.editStep(${index})">編集</button>
1672
+ <button type="button" class="btn btn-danger btn-sm" onclick="StepBuilder.deleteStep(${index})">削除</button>
1673
+ </div>
1674
+ </li>
1675
+ `;
1676
+ }).join('');
1677
+
1678
+ initSortable();
1679
+ }
1680
+
1681
+ function getFieldLabels(fieldNames) {
1682
+ if (!fieldNames || fieldNames.length === 0) return '';
1683
+ const fields = typeof FormBuilder !== 'undefined' ? FormBuilder.getFields() : [];
1684
+ return fieldNames.map(name => {
1685
+ const field = fields.find(f => f.name === name);
1686
+ return field ? (field.label || field.name) : name;
1687
+ }).join(', ');
1688
+ }
1689
+
1690
+ function escapeHtml(str) {
1691
+ if (!str) return '';
1692
+ return str.replace(/&/g, '&amp;')
1693
+ .replace(/</g, '&lt;')
1694
+ .replace(/>/g, '&gt;')
1695
+ .replace(/"/g, '&quot;');
1696
+ }
1697
+
1698
+ function save() {
1699
+ const input = document.getElementById('steps');
1700
+ if (input) {
1701
+ input.value = JSON.stringify(steps);
1702
+ }
1703
+ }
1704
+
1705
+ function showAddModal() {
1706
+ document.getElementById('step-modal-title').textContent = 'ステップ追加';
1707
+ document.getElementById('step-edit-index').value = '-1';
1708
+ document.getElementById('step-title').value = '';
1709
+ renderFieldSelector([]);
1710
+ document.getElementById('step-modal').style.display = 'flex';
1711
+ }
1712
+
1713
+ function editStep(index) {
1714
+ const step = steps[index];
1715
+ if (!step) return;
1716
+
1717
+ document.getElementById('step-modal-title').textContent = 'ステップ編集';
1718
+ document.getElementById('step-edit-index').value = index;
1719
+ document.getElementById('step-title').value = step.title || '';
1720
+ renderFieldSelector(step.fields || []);
1721
+ document.getElementById('step-modal').style.display = 'flex';
1722
+ }
1723
+
1724
+ function renderFieldSelector(selectedFields) {
1725
+ const selector = document.getElementById('step-fields-selector');
1726
+ const noFields = document.getElementById('step-no-fields');
1727
+ const fields = typeof FormBuilder !== 'undefined' ? FormBuilder.getFields() : [];
1728
+
1729
+ if (fields.length === 0) {
1730
+ selector.innerHTML = '<p class="text-muted">先にフィールドを追加してください</p>';
1731
+ return;
1732
+ }
1733
+
1734
+ selector.innerHTML = fields.map(field => {
1735
+ const checked = selectedFields.includes(field.name) ? 'checked' : '';
1736
+ return `
1737
+ <label class="step-field-option">
1738
+ <input type="checkbox" value="${escapeHtml(field.name)}" ${checked}>
1739
+ <span>${escapeHtml(field.label || field.name)}</span>
1740
+ </label>
1741
+ `;
1742
+ }).join('');
1743
+ }
1744
+
1745
+ function hideModal() {
1746
+ document.getElementById('step-modal').style.display = 'none';
1747
+ }
1748
+
1749
+ function saveStep() {
1750
+ const title = document.getElementById('step-title').value.trim();
1751
+ if (!title) {
1752
+ alert('ステップタイトルは必須です');
1753
+ return;
1754
+ }
1755
+
1756
+ const editIndex = parseInt(document.getElementById('step-edit-index').value);
1757
+
1758
+ // Collect selected fields
1759
+ const selectedFields = [];
1760
+ document.querySelectorAll('#step-fields-selector input[type="checkbox"]:checked').forEach(cb => {
1761
+ selectedFields.push(cb.value);
1762
+ });
1763
+
1764
+ const step = {
1765
+ title: title,
1766
+ fields: selectedFields
1767
+ };
1768
+
1769
+ if (editIndex >= 0) {
1770
+ steps[editIndex] = step;
1771
+ } else {
1772
+ steps.push(step);
1773
+ }
1774
+
1775
+ save();
1776
+ render();
1777
+ hideModal();
1778
+ }
1779
+
1780
+ function deleteStep(index) {
1781
+ if (!confirm('このステップを削除しますか?')) return;
1782
+ steps.splice(index, 1);
1783
+ save();
1784
+ render();
1785
+ }
1786
+
1787
+ // Initialize on DOM ready
1788
+ document.addEventListener('DOMContentLoaded', init);
1789
+
1790
+ // Close modal on escape
1791
+ document.addEventListener('keydown', function(e) {
1792
+ if (e.key === 'Escape') {
1793
+ const modal = document.getElementById('step-modal');
1794
+ if (modal && modal.style.display === 'flex') {
1795
+ hideModal();
1796
+ }
1797
+ }
1798
+ });
1799
+
1800
+ return {
1801
+ showAddModal,
1802
+ editStep,
1803
+ deleteStep,
1804
+ hideModal,
1805
+ saveStep
1460
1806
  };
1461
1807
  })();
1462
1808
  </script>
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: focomy
3
- Version: 0.1.105
3
+ Version: 0.1.106
4
4
  Summary: The Most Beautiful CMS - A metadata-driven, zero-duplicate-code content management system
5
5
  Project-URL: Homepage, https://github.com/focomy/focomy
6
6
  Project-URL: Documentation, https://focomy.dev/docs
@@ -2,7 +2,7 @@ core/__init__.py,sha256=SHwUSW_pu27kEARvwJFRdY2ZkVu6wzFh0eikYJXjbTY,1123
2
2
  core/cli.py,sha256=h5jnf4EJCZ1JZMnDl7W1IaL153UD3dXyKBbxhv1P-KA,35536
3
3
  core/config.py,sha256=kLhM1yF3rU9KtJ-DiD8pnzG_tFQsiO8G-M_ffN_yKqE,8200
4
4
  core/database.py,sha256=wqvRuwpDxy25svag9NqreVcNA_VRDDjWsxOMUopsffs,2222
5
- core/main.py,sha256=1R5NH9rxBSApaMM6CxJszf0hJE9o_bfXgKOXHkGjOJM,16765
5
+ core/main.py,sha256=em4na_OJu6E3y_nkeYxH_QvugiLdPzKCMbc5Jbc1y1k,16783
6
6
  core/rate_limit.py,sha256=CX5UjmsU03aFWKXSKjweoHvH2xn0v4NBHNN5ynJC8LE,180
7
7
  core/relations.yaml,sha256=7GUCrphKaouEXNkyd8Ht99e6TcUPERhc4m36RGcc41U,2128
8
8
  core/utils.py,sha256=7myy64jI7T4WQ_C3Q7j0RPaYKQhfVzGkpNrw9c4yakc,1203
@@ -26,7 +26,7 @@ core/content_types/category.yaml,sha256=NoRTdksopCb3JbQocCKHVOdPr4cMkUzkgParB9HH
26
26
  core/content_types/channel.yaml,sha256=Li6WlpZJqnHYJnGRhr7CLXYRq5sqspxPxGgQpSeSRWc,513
27
27
  core/content_types/comment.yaml,sha256=XUpfNc6k6fMYXPuwsu8XQVHoy57H7oFeEBQRnnJLa9A,1031
28
28
  core/content_types/edit_lock.yaml,sha256=PDavL9Mor0qyMz0l97v59L9XBHa0ZINhs05kQ2WNW2c,599
29
- core/content_types/form.yaml,sha256=kdfL00kPAtPApoluobyGEfFMUcJ0BFB1M0eAXH4PbZU,1782
29
+ core/content_types/form.yaml,sha256=YP8SfWzAEJnhJ9Ifa0gNtY4YlR0N07dCyojRhW0y6Y0,1432
30
30
  core/content_types/form_submission.yaml,sha256=vVJ-5JA-EJ9--QxvXSm0JwVkjXdUQcw5iYKj5_LgLf4,810
31
31
  core/content_types/menu_item.yaml,sha256=JgCQtXN5aZvaarFLv6tU74rjr01CmsdXDO9SHI8RyYA,1449
32
32
  core/content_types/news.yaml,sha256=Pykllr3t9E5zooQn41ZtR9xeUj0AbfOZCeoGnFQFH7E,1017
@@ -153,7 +153,7 @@ core/templates/admin/base.html,sha256=yWQNumbtcLIE0iffIoDB2xTF-9W5bt2mdsMYqA5QOa
153
153
  core/templates/admin/comments.html,sha256=AEzCMYu1-EdFC1yUa9oIObtRS-U9RlMTe_5P3O5alb8,10012
154
154
  core/templates/admin/customize.html,sha256=EwncOC0AgDCSZRum5PZzF-KDYNjrhPish57KMuff4sE,12260
155
155
  core/templates/admin/dashboard.html,sha256=GsYHM4OYLZWCmleeHrtyS0g8zKhFmlnacJg1LcfGzxM,2513
156
- core/templates/admin/entity_form.html,sha256=EPcQrc-awntjoYeRCd7vKbQQ8799kKgEG4E56-aCBy0,48721
156
+ core/templates/admin/entity_form.html,sha256=7LuCyjliuc7bk6S_5Nn4OcXMCfULgZ2z8H48FR88zPo,60261
157
157
  core/templates/admin/entity_list.html,sha256=DxaM2c_1UGdPpQHTIe7LrFxjC11f8hXDy-9C-T8LO6o,7966
158
158
  core/templates/admin/forgot_password.html,sha256=j2r9N5G8T5zKODQu7IVc0NdRvSVuyXBA9FCLn7egp1Q,2320
159
159
  core/templates/admin/import.html,sha256=2b2Bu9Do1pQBUgsseowt19oaf0vEfGzTOCJ8Rr-Nphc,48385
@@ -201,8 +201,8 @@ themes/minimal/templates/base.html,sha256=LFkx-XLDMGH7oFHHa0e6KPB0DJITOBvr6GtPkD
201
201
  themes/minimal/templates/home.html,sha256=ygYQgYj1OGCiKwmfsxwkPselVKT8vDH3jLLbfphpqKI,1577
202
202
  themes/minimal/templates/page.html,sha256=7Xcoq-ryaxlp913H2S1ishrAro2wsqqGmvsm1osXxd4,389
203
203
  themes/minimal/templates/post.html,sha256=FkTRHci8HNIIi3DU6Mb3oL0aDisGyDcsT_IUDwHmrvo,1387
204
- focomy-0.1.105.dist-info/METADATA,sha256=Sluq3xJdDBPFc3_VeQoGE9rQv0PgFdzO6dOmNBuL1V8,7042
205
- focomy-0.1.105.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
206
- focomy-0.1.105.dist-info/entry_points.txt,sha256=_rF-wxGI1axY7gox3DBsTLHq-JrFKkMCjA65a6b_oqE,41
207
- focomy-0.1.105.dist-info/licenses/LICENSE,sha256=z9Z7gN7NNV7zYCaY-Knh3bv8RBCu89VueYtAlN_-lro,1063
208
- focomy-0.1.105.dist-info/RECORD,,
204
+ focomy-0.1.106.dist-info/METADATA,sha256=WuqHmxR0zsPOR2cyZW0vV6yIEBkUgIneDCm1xXgZ30U,7042
205
+ focomy-0.1.106.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
206
+ focomy-0.1.106.dist-info/entry_points.txt,sha256=_rF-wxGI1axY7gox3DBsTLHq-JrFKkMCjA65a6b_oqE,41
207
+ focomy-0.1.106.dist-info/licenses/LICENSE,sha256=z9Z7gN7NNV7zYCaY-Knh3bv8RBCu89VueYtAlN_-lro,1063
208
+ focomy-0.1.106.dist-info/RECORD,,