yummy-guide-generic-administrate 0.3.1 → 0.4.0

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b442ee0d24ba8d34271193cdbc8c2da175d5b91b3a2b847706ce5f5c316eff66
4
- data.tar.gz: cc6d60ef4a0b80f2561638de33ce7abef3908ba1064c77c2c94e628d47dec970
3
+ metadata.gz: 7a900e85ac77c86b162da45de406b17c80f1aa18340da95f6d02ac45ca8ce259
4
+ data.tar.gz: 80d5fca1e0d8248f26906395613ff1974b0704d306cece30985f2ebf954a7b67
5
5
  SHA512:
6
- metadata.gz: ad2200cef92ab686ffb9fabee36c87672ff48dc1a507406995057effa3afff4bad625eb7fbaad8011ddf4ffa8eb93b5b95af2455436f131fabbdc18a8ff55873
7
- data.tar.gz: 2502a4c2723a4d8fc25ca7d1ba5fd10aff5a2002dbfff58f5c698d042b21e21ebd0597c863463ec1b086934b3450e56088e718d5fddae8a8b1a861bd02bf08c5
6
+ metadata.gz: 105bcc475d2ac342e51ff9765dda3ca8e3709949a1b72d471182ec90c6a3df1fe17bc014a09bce4ef27f4976e0dddc41d2457ea4b2b8424345a5cb4a24c69cfe
7
+ data.tar.gz: afca6ea454ec4ddb804234954df68ab30790a58bc109a8555afaab9d012fd480aed5c3cff3d749cd5ca001851d10a0d788f82180821877f60be0ef83fcd1d155
data/README.md CHANGED
@@ -151,7 +151,7 @@ gem には専用 partial があります。
151
151
  <%= render "yummy_guide/administrate/administrate/application/fixed_table_header" %>
152
152
  </header>
153
153
 
154
- <section class="main-content__body">
154
+ <section class="main-content__body main-content__body--flush">
155
155
  <%= render "yummy_guide/administrate/administrate/application/collection",
156
156
  collection_presenter: collection_presenter,
157
157
  page: page,
@@ -166,11 +166,12 @@ gem には専用 partial があります。
166
166
  `fixed_table_header` partial 自体は以下の 1 行です。
167
167
 
168
168
  ```erb
169
- <div data-fixed-table-header hidden></div>
169
+ <div class="yummy-guide-administrate-fixed-table-header" data-fixed-table-header hidden></div>
170
170
  ```
171
171
 
172
172
  この slot を置かない場合でも、JS が table の直前に自動生成します。配置を制御
173
- したいときだけ明示してください。
173
+ したいときだけ明示してください。desktop では LMJ と同じく、`main-content__body--flush`
174
+ と組み合わせることで fixed header のクリップ幅が一覧 body と揃います。
174
175
 
175
176
  #### 3. 自前の table partial を使う場合
176
177
 
@@ -3,6 +3,7 @@
3
3
  var WRAPPER_SELECTOR = '[data-fixed-header-scroll], .scroll-table, .home-table__wrapper, .table-wrap, .af__table__content';
4
4
  var resizeObservers = new WeakMap();
5
5
  var observedScrolls = new WeakMap();
6
+ var observedFixedScrolls = new WeakMap();
6
7
  var scrollSyncStates = new WeakMap();
7
8
 
8
9
  function directCells(row) {
@@ -232,10 +233,61 @@
232
233
  if (!scroll || !fixedScroll) return;
233
234
 
234
235
  var left = typeof scrollLeft === 'number' ? scrollLeft : scroll.scrollLeft;
236
+ var state = ensureScrollSyncState(scroll);
237
+ var currentLeft = preciseNumber(fixedScroll.scrollLeft);
235
238
 
239
+ if (Math.abs(currentLeft - left) < 0.5) {
240
+ state.ignoredFixedScrollLeft = null;
241
+ return;
242
+ }
243
+
244
+ state.ignoredFixedScrollLeft = preciseNumber(left);
236
245
  fixedScroll.scrollLeft = left;
237
246
  }
238
247
 
248
+ function syncSourceScroll(scroll, fixedScroll, scrollLeft) {
249
+ if (!scroll || !fixedScroll) return;
250
+
251
+ var left = typeof scrollLeft === 'number' ? scrollLeft : fixedScroll.scrollLeft;
252
+ var state = ensureScrollSyncState(scroll);
253
+ var currentLeft = preciseNumber(scroll.scrollLeft);
254
+
255
+ if (Math.abs(currentLeft - left) < 0.5) {
256
+ state.ignoredSourceScrollLeft = null;
257
+ return;
258
+ }
259
+
260
+ state.ignoredSourceScrollLeft = preciseNumber(left);
261
+ scroll.scrollLeft = left;
262
+ }
263
+
264
+ function ensureScrollSyncState(scroll) {
265
+ var state = scrollSyncStates.get(scroll);
266
+
267
+ if (state) return state;
268
+
269
+ state = {
270
+ sourceFrame: null,
271
+ fixedFrame: null,
272
+ sourceScrollLeft: 0,
273
+ fixedScrollLeft: 0,
274
+ ignoredSourceScrollLeft: null,
275
+ ignoredFixedScrollLeft: null
276
+ };
277
+
278
+ scrollSyncStates.set(scroll, state);
279
+ return state;
280
+ }
281
+
282
+ function consumeIgnoredScrollLeft(state, key, scrollLeft) {
283
+ if (!state || state[key] === null || typeof state[key] === 'undefined') return false;
284
+
285
+ var ignoredLeft = state[key];
286
+ state[key] = null;
287
+
288
+ return Math.abs(ignoredLeft - preciseNumber(scrollLeft)) < 0.5;
289
+ }
290
+
239
291
  function fixedHeaderTableClassName(sourceTable) {
240
292
  var excludedClasses = {
241
293
  'table-with-fixed-header': true,
@@ -508,35 +560,57 @@
508
560
  });
509
561
  }
510
562
 
511
- function observeFixedHeaderScroll(scroll) {
563
+ function observeSourceScroll(scroll) {
512
564
  if (observedScrolls.has(scroll)) return;
513
565
 
514
566
  scroll.addEventListener('scroll', function() {
515
- var state = scrollSyncStates.get(scroll) || {};
567
+ var state = ensureScrollSyncState(scroll);
568
+ if (consumeIgnoredScrollLeft(state, 'ignoredSourceScrollLeft', scroll.scrollLeft)) return;
569
+
516
570
  state.scrollLeft = scroll.scrollLeft;
571
+ state.sourceScrollLeft = scroll.scrollLeft;
517
572
 
518
- if (state.frame) return;
573
+ if (state.sourceFrame) return;
519
574
 
520
- state.frame = window.requestAnimationFrame(function() {
575
+ state.sourceFrame = window.requestAnimationFrame(function() {
521
576
  var slot = ensureFixedHeaderSlot(scroll);
522
- state.frame = null;
523
- syncFixedHeaderScroll(scroll, slot.querySelector('.table-fixed-header__scroll'), state.scrollLeft);
577
+ state.sourceFrame = null;
578
+ syncFixedHeaderScroll(scroll, ensureFixedHeaderScrollViewport(slot), state.sourceScrollLeft);
524
579
  });
525
-
526
- scrollSyncStates.set(scroll, state);
527
580
  }, { passive: true });
528
581
 
529
582
  observedScrolls.set(scroll, true);
530
583
  }
531
584
 
585
+ function observeFixedHeaderViewport(scroll, fixedScroll) {
586
+ if (!fixedScroll || observedFixedScrolls.get(fixedScroll) === scroll) return;
587
+
588
+ fixedScroll.addEventListener('scroll', function() {
589
+ var state = ensureScrollSyncState(scroll);
590
+ if (consumeIgnoredScrollLeft(state, 'ignoredFixedScrollLeft', fixedScroll.scrollLeft)) return;
591
+
592
+ state.fixedScrollLeft = fixedScroll.scrollLeft;
593
+
594
+ if (state.fixedFrame) return;
595
+
596
+ state.fixedFrame = window.requestAnimationFrame(function() {
597
+ state.fixedFrame = null;
598
+ syncSourceScroll(scroll, fixedScroll, state.fixedScrollLeft);
599
+ });
600
+ }, { passive: true });
601
+
602
+ observedFixedScrolls.set(fixedScroll, scroll);
603
+ }
604
+
532
605
  function initializeFixedHeaderForScroll(scroll) {
533
606
  var sourceTable = activeSourceTable(scroll);
534
607
  if (!sourceTable) return;
535
608
 
536
609
  var slot = ensureFixedHeaderSlot(scroll);
537
610
  buildFixedTableHeader(slot, scroll, sourceTable);
611
+ observeFixedHeaderViewport(scroll, slot.querySelector('.table-fixed-header__scroll'));
538
612
  observeFixedHeader(scroll);
539
- observeFixedHeaderScroll(scroll);
613
+ observeSourceScroll(scroll);
540
614
  }
541
615
 
542
616
  function initializeFixedHeaders(root) {
@@ -1,5 +1,79 @@
1
1
  @import "fixed_submit_actions";
2
2
 
3
+ @media screen and (min-width: 768px) {
4
+ .app-container {
5
+ --app-container-block-padding: 1.5em;
6
+ align-items: flex-start;
7
+ gap: 1.5em;
8
+ min-height: 100vh;
9
+ min-height: 100dvh;
10
+ height: auto;
11
+ overflow: visible;
12
+ box-sizing: border-box;
13
+ margin-left: 0;
14
+ margin-right: 0;
15
+ max-width: 100%;
16
+ padding: var(--app-container-block-padding) 1.5em;
17
+ }
18
+
19
+ .navigation {
20
+ display: flex;
21
+ flex: 0 0 240px;
22
+ flex-direction: column;
23
+ width: 240px;
24
+ max-width: 240px;
25
+ height: auto;
26
+ max-height: calc(100vh - 3em);
27
+ max-height: calc(100dvh - 3em);
28
+ overflow-y: auto;
29
+ overscroll-behavior: contain;
30
+ position: sticky;
31
+ top: 1.5em;
32
+ align-self: flex-start;
33
+ }
34
+
35
+ .navigation .navigation__link__secondary {
36
+ margin-left: 0;
37
+ }
38
+
39
+ .main-content {
40
+ flex: 1 1 auto;
41
+ min-width: 0;
42
+ min-height: calc(100vh - (var(--app-container-block-padding) + var(--app-container-block-padding)));
43
+ min-height: calc(100dvh - (var(--app-container-block-padding) + var(--app-container-block-padding)));
44
+ height: auto;
45
+ overflow: visible;
46
+ margin-bottom: 0;
47
+ padding-bottom: 1rem;
48
+ }
49
+
50
+ .main-content__header--sticky-table-layout {
51
+ position: sticky;
52
+ top: 0;
53
+ z-index: 10;
54
+ flex-wrap: wrap;
55
+ overflow-x: hidden;
56
+ background: #fff;
57
+ }
58
+
59
+ .main-content__body--flush {
60
+ padding: 0 0.5em 1rem;
61
+ }
62
+
63
+ .table-fixed-header-slot[data-fixed-header-inline="true"] {
64
+ flex-basis: calc(100% + 4rem - 1em);
65
+ flex-shrink: 0;
66
+ margin: 0 calc(-2rem + 0.5em) -1rem;
67
+ overflow-x: hidden;
68
+ background: #fff;
69
+ }
70
+
71
+ .table-fixed-header-slot[data-fixed-header-inline="true"] .home-table--fixed-header {
72
+ min-width: 100%;
73
+ margin: 0;
74
+ }
75
+ }
76
+
3
77
  .scroll-table {
4
78
  overflow-x: auto;
5
79
  }
@@ -419,6 +493,7 @@ td.actions-column .sticky__item,
419
493
  word-break: break-word;
420
494
  }
421
495
 
496
+
422
497
  .home-table--fixed-header th.sticky {
423
498
  position: sticky;
424
499
  right: 0;
@@ -2,6 +2,6 @@
2
2
 
3
3
  module YummyGuide
4
4
  module Administrate
5
- VERSION = "0.3.1"
5
+ VERSION = "0.4.0"
6
6
  end
7
7
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: yummy-guide-generic-administrate
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.3.1
4
+ version: 0.4.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - akatsuki-kk