katalyst-tables 3.5.0 → 3.5.2
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 +4 -4
- data/README.md +5 -262
- data/app/assets/builds/katalyst/tables.esm.js +1 -1
- data/app/assets/builds/katalyst/tables.js +1 -1
- data/app/assets/builds/katalyst/tables.min.js +1 -1
- data/app/assets/builds/katalyst/tables.min.js.map +1 -1
- data/app/assets/stylesheets/katalyst/tables/_ordinal.scss +1 -1
- data/app/assets/stylesheets/katalyst/tables/_query.scss +5 -3
- data/app/components/katalyst/tables/pagy_nav_component.rb +15 -1
- data/app/components/katalyst/tables/query/modal_component.rb +6 -0
- data/app/components/katalyst/tables/selectable/form_component.html.erb +1 -7
- data/app/components/katalyst/tables/selectable/form_component.rb +14 -1
- data/app/controllers/concerns/katalyst/tables/backend.rb +14 -0
- data/app/helpers/katalyst/tables/frontend.rb +13 -0
- data/app/javascript/tables/query_controller.js +1 -1
- data/app/models/concerns/katalyst/tables/collection/pagination.rb +6 -6
- data/lib/katalyst/tables/collection/type/enum.rb +0 -17
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 5402c4de7c04e40f7586c7ab01498e17a085b16c630a556eecbd5ddf7557fc08
|
4
|
+
data.tar.gz: 52de56d41efe1dc78a47e43891db23bcca41feccb285f450e4fb45e5810d51f0
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 107ebb571430631da2083805322b3734d0a1ef5ca9f4bbf43433e0e32658ea3fec5ad8bb94e8cf768978e2b1d9224f755f81d0e32b69e59737425f23c5f9c68f
|
7
|
+
data.tar.gz: c43331d1a1fc21587f0cd5a874a5adbf34813b94eae3007586418b3c440876ec00a8e810aa19eb2d15eb52b55f4c37280dee8992f8e86e95ae826094bfc5ad74
|
data/README.md
CHANGED
@@ -1,267 +1,10 @@
|
|
1
|
-
# Katalyst
|
1
|
+
# Katalyst Tables
|
2
2
|
|
3
|
-
|
3
|
+
A library for viewing and interacting with database tables in Ruby on Rails.
|
4
4
|
|
5
|
-
##
|
5
|
+
## Documentation
|
6
6
|
|
7
|
-
|
8
|
-
|
9
|
-
```ruby
|
10
|
-
gem "katalyst-tables"
|
11
|
-
```
|
12
|
-
|
13
|
-
And then execute:
|
14
|
-
|
15
|
-
$ bundle install
|
16
|
-
|
17
|
-
Add the Gem's javascript and CSS to your build pipeline. This assumes that
|
18
|
-
you're using `rails-dartsass` and `importmaps` to manage your assets.
|
19
|
-
|
20
|
-
```javascript
|
21
|
-
// app/javascript/controllers/application.js
|
22
|
-
import { application } from "controllers/application";
|
23
|
-
import tables from "@katalyst/tables";
|
24
|
-
application.load(tables);
|
25
|
-
```
|
26
|
-
|
27
|
-
```scss
|
28
|
-
// app/assets/stylesheets/application.scss
|
29
|
-
@use "@katalyst/tables";
|
30
|
-
```
|
31
|
-
|
32
|
-
## Usage
|
33
|
-
|
34
|
-
This gem provides entry points for backend and frontend concerns:
|
35
|
-
* `Katalyst::TableComponent` can be used render encapsulated tables,
|
36
|
-
* `Katalyst::SummaryTableComponent` can be used render a record using the table syntax,
|
37
|
-
* `Katalyst::Tables::Frontend` provides `table_with` for inline table generation,
|
38
|
-
* `Katalyst::Tables::Collection::Base` provides a default entry point for
|
39
|
-
building collections in your controller actions
|
40
|
-
* `Katalyst::Tables::Collection::Query` provides build-in query parsing and filtering
|
41
|
-
based on the attributes defined in your collection.
|
42
|
-
|
43
|
-
### Frontend
|
44
|
-
|
45
|
-
Add `include Katalyst::Tables::Frontend` to your `ApplicationHelper` or similar.
|
46
|
-
|
47
|
-
You can use the `table_with` helper to generate a table inline in your view without explicitly interacting with the
|
48
|
-
table component. This is the preferred approach when creating tables. However, if the table is complex or you need to
|
49
|
-
reuse it, you should consider moving the definition of the row into a partial.
|
50
|
-
|
51
|
-
```erb
|
52
|
-
<%= table_with collection: @people do |row, person| %>
|
53
|
-
<% row.text :name do |cell| %>
|
54
|
-
<%= link_to cell.value, [:edit, person] %>
|
55
|
-
<% end %>
|
56
|
-
<% row.text :email %>
|
57
|
-
<% end %>
|
58
|
-
```
|
59
|
-
|
60
|
-
By not providing a block to the `table_with` call, the gem will look for a partial called `_person.html+row.erb` to
|
61
|
-
render each row:
|
62
|
-
|
63
|
-
```erb
|
64
|
-
<%# locals: { row:, person: nil } %>
|
65
|
-
<% row.text :name do |cell| %>
|
66
|
-
<%= link_to cell.value, [:edit, person] %>
|
67
|
-
<% end %>
|
68
|
-
<% row.text :email %>
|
69
|
-
```
|
70
|
-
|
71
|
-
The table will call your block / partial once per row and accumulate the cells
|
72
|
-
you generate into rows, including a header row:
|
73
|
-
|
74
|
-
```html
|
75
|
-
|
76
|
-
<table>
|
77
|
-
<thead>
|
78
|
-
<tr>
|
79
|
-
<th>Name</th>
|
80
|
-
<th>Email</th>
|
81
|
-
</tr>
|
82
|
-
</thead>
|
83
|
-
<tbody>
|
84
|
-
<tr>
|
85
|
-
<td><a href="/people/1/edit">Alice</a></td>
|
86
|
-
<td>alice@acme.org</td>
|
87
|
-
</tr>
|
88
|
-
<tr>
|
89
|
-
<td><a href="/people/2/edit">Bob</a></td>
|
90
|
-
<td>bob@acme.org</td>
|
91
|
-
</tr>
|
92
|
-
</tbody>
|
93
|
-
</table>
|
94
|
-
```
|
95
|
-
|
96
|
-
You can customize the partial and/or the name of the resource in a similar style
|
97
|
-
to view partials:
|
98
|
-
|
99
|
-
```erb
|
100
|
-
<%= table_with(collection: @employees, as: :person, partial: "person") %>
|
101
|
-
```
|
102
|
-
|
103
|
-
### HTML Attributes
|
104
|
-
|
105
|
-
You can add custom attributes on table, row, and cell tags.
|
106
|
-
|
107
|
-
The table tag takes attributes passed to `table_with` helper, similar to `form_with`:
|
108
|
-
|
109
|
-
```erb
|
110
|
-
<%= table_with(collection: @people, id: "people-table")
|
111
|
-
```
|
112
|
-
|
113
|
-
Cells support the same approach:
|
114
|
-
|
115
|
-
```erb
|
116
|
-
<%= row.text :name, class: "name" %>
|
117
|
-
```
|
118
|
-
|
119
|
-
Rows do not get called directly, so instead you can assign to `html_attributes` on the row builder to customize row tag
|
120
|
-
generation.
|
121
|
-
|
122
|
-
```erb
|
123
|
-
<% row.update_html_attributes(id: person.id) if row.body? %>
|
124
|
-
```
|
125
|
-
|
126
|
-
Note: because the row builder gets called to generate the header row, you may need to guard calls that access the
|
127
|
-
`person` directly as shown in the previous example. You could also check whether `person` is present.
|
128
|
-
|
129
|
-
### Headers
|
130
|
-
|
131
|
-
Tables will automatically generate a header row for you by calling your row partial or provided block with no object.
|
132
|
-
During this call, `row.header?` is true, `row.body?` is false, and the object (`person`) is nil.
|
133
|
-
|
134
|
-
All cells generated in the table header iteration will automatically be header cells, but you can also make header cells
|
135
|
-
in your body rows by passing `heading: true` when you generate the cell.
|
136
|
-
|
137
|
-
```erb
|
138
|
-
<% row.number :id, heading: true %>
|
139
|
-
```
|
140
|
-
|
141
|
-
The table header cells default to showing the capitalized column name, but you can customize this in one of two ways:
|
142
|
-
|
143
|
-
* Set the value inline
|
144
|
-
```erb
|
145
|
-
<% row.number :id, label: "ID" %>
|
146
|
-
```
|
147
|
-
* Define a translation for the attribute
|
148
|
-
```yml
|
149
|
-
# en.yml
|
150
|
-
activerecord:
|
151
|
-
attributes:
|
152
|
-
person:
|
153
|
-
id: "ID"
|
154
|
-
```
|
155
|
-
|
156
|
-
Note: if the cell is given a block, it is not called during the header pass. This
|
157
|
-
is because the block is assumed to be for generating data for the body, not the
|
158
|
-
header. We suggest you set `label` instead.
|
159
|
-
|
160
|
-
#### Cell values
|
161
|
-
|
162
|
-
If you do not provide a value when you call the cell builder, the attribute you
|
163
|
-
provide will be retrieved from the current item and the result will be rendered in
|
164
|
-
the table cell. This is often all you need to do, but if you do want to customise
|
165
|
-
the value you can pass a block instead:
|
166
|
-
|
167
|
-
```erb
|
168
|
-
<% row.text :status do %>
|
169
|
-
<%= person.password.present? ? "Active" : "Invited" %>
|
170
|
-
<% end %>
|
171
|
-
```
|
172
|
-
|
173
|
-
In the context of the block you have access the cell component if you simply
|
174
|
-
want to extend the default behaviour:
|
175
|
-
|
176
|
-
```erb
|
177
|
-
<%# @type [Katalyst::Tables::CellComponent] cell %>
|
178
|
-
<% row.text :name do |cell| %>
|
179
|
-
<%= link_to cell, person %>
|
180
|
-
<% end %>
|
181
|
-
```
|
182
|
-
|
183
|
-
You can also update `html_attributes` on the cell builder, similar to the row
|
184
|
-
builder, see `katalyst-html-attributes` for details.
|
185
|
-
|
186
|
-
## Collections
|
187
|
-
|
188
|
-
The `Katalyst::Tables::Collection::Base` class provides a convenient way to
|
189
|
-
manage collections in your controller actions. It is designed to be used with
|
190
|
-
Pagy for pagination and provides built-in sorting when used with ActiveRecord
|
191
|
-
collections. Sorting and Pagination are off by default, you can either set them
|
192
|
-
on creation or create a custom `Collection` class that sets them on by default:
|
193
|
-
|
194
|
-
```ruby
|
195
|
-
# in #index
|
196
|
-
Katalyst::Tables::Collection::Base.new(sorting: "name asc", pagination: true)
|
197
|
-
# or as a nested class in your controller
|
198
|
-
class Collection < Katalyst::Tables::Collection::Base
|
199
|
-
config.sorting = "name asc" # requires models have a name attribute
|
200
|
-
config.pagination = true
|
201
|
-
end
|
202
|
-
```
|
203
|
-
|
204
|
-
Collections can be passed directly to `table_with` method and it will automatically
|
205
|
-
detect features such as sorting and generate the appropriate table header links.
|
206
|
-
|
207
|
-
```erb
|
208
|
-
<%= table_with(collection:) %>
|
209
|
-
```
|
210
|
-
|
211
|
-
### Query
|
212
|
-
|
213
|
-
Include `Katalyst::Tables::Collection::Query` into your collection to add automatic
|
214
|
-
query parsing and filtering based on the configured attributes. For example:
|
215
|
-
|
216
|
-
```ruby
|
217
|
-
class Collection < Katalyst::Tables::Collection::Base
|
218
|
-
include Katalyst::Tables::Collection::Query
|
219
|
-
|
220
|
-
attribute :first_name, :string
|
221
|
-
attribute :created_at, :date
|
222
|
-
end
|
223
|
-
```
|
224
|
-
|
225
|
-
With this definition and a text-input named `query`, your users can write tagged
|
226
|
-
query expressions such as `first_name:Aaron` or `created_at:>2024-01-01` and these
|
227
|
-
will be automatically parsed and applied to the collection attribute, and the collection
|
228
|
-
will automatically generate and apply ActiveRecord conditions to filter the given scope.
|
229
|
-
|
230
|
-
There's also a frontend utility, `table_query_with(collection:)` that will generate the form
|
231
|
-
and show a modal that helps users to interact with the query interface. More details available
|
232
|
-
in the [query](docs/query.md) documentation.
|
233
|
-
|
234
|
-
## Summary tables
|
235
|
-
You can use the `Katalyst::SummaryTableComponent` to render a single record utilizing all the functionality from the
|
236
|
-
`Katalyst::TableComponent`.
|
237
|
-
|
238
|
-
```erb
|
239
|
-
<%= summary_table_with model: @person do |row| %>
|
240
|
-
<% row.text :name %>
|
241
|
-
<% row.text :email %>
|
242
|
-
<% end %>
|
243
|
-
```
|
244
|
-
|
245
|
-
## Extensions
|
246
|
-
|
247
|
-
The following extensions are available and activated by default:
|
248
|
-
|
249
|
-
* [Filtering](docs/filtering.md) - adds automatic collection filtering based on attributes
|
250
|
-
* [Query](docs/query.md) - adds human-friendly text filtering that populates collection attributes
|
251
|
-
* [Identifiable](docs/identifiable.md) - adds default dom ids to the table and data rows.
|
252
|
-
* [Orderable](docs/orderable.md) - adds bulk-update for 'ordinal' columns via dragging rows in the table.
|
253
|
-
* [Pagination](docs/pagination.md) - handles paginating of data in the collection.
|
254
|
-
* [Selectable](docs/selectable.md) - adds bulk-action support for rows in the table.
|
255
|
-
* [Sortable](docs/sortable.md) - table column headers that can be sorted will be wrapped in links.
|
256
|
-
* [Customization](docs/customization.md) - customize the table and cell rendering.
|
257
|
-
|
258
|
-
You can disable extensions by altering the `Katalyst::Tables.config.component_extensions` before initialization.
|
259
|
-
|
260
|
-
## Development
|
261
|
-
|
262
|
-
After checking out the repo, run `bin/setup` to install dependencies. Then, run `bundle exec rspec` to run the tests.
|
263
|
-
|
264
|
-
To install this gem onto your local machine, run `bundle exec rake install`.
|
7
|
+
See [katalyst.github.io/tables](https://katalyst.github.io/tables/) for documentation.
|
265
8
|
|
266
9
|
## Contributing
|
267
10
|
|
@@ -269,4 +12,4 @@ Bug reports and pull requests are welcome on GitHub at https://github.com/kataly
|
|
269
12
|
|
270
13
|
## License
|
271
14
|
|
272
|
-
|
15
|
+
Katalyst Tables is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
|
@@ -1,2 +1,2 @@
|
|
1
|
-
import{Controller as e}from"@hotwired/stimulus";class t{constructor(e,t,s){this.cursorOffset=t.offsetY,this.initialPosition=t.target.offsetTop-e.offsetTop,this.targetId=s}updateCursor(e,t,s,i){this.listOffset=e.getBoundingClientRect().top;let n=s.clientY-this.listOffset-this.cursorOffset;this.#e(e,t,n,i)}updateScroll(e,t,s){const i=this.listOffset;this.listOffset=e.getBoundingClientRect().top;const n=i-this.listOffset,r=this.position+n;this.#e(e,t,r,s)}#e(e,t,s,i){s=Math.max(s,0),s=Math.min(s,e.offsetHeight-t.offsetHeight),this.position=s;i(s-this.initialPosition)}}class s extends e{static outlets=["tables--selection--form"];static values={params:Object,checked:Boolean};tablesSelectionFormOutletConnected(e){e.visible(this.id,!0),this.checkedValue=e.isSelected(this.id)}disconnect(){this.hasTablesSelectionFormOutlet&&this.tablesSelectionFormOutlet.visible(this.id,!1)}change(e){e.preventDefault(),this.checkedValue=this.tablesSelectionFormOutlet.toggle(this.id)}get id(){return this.paramsValue.id}paramsValueChanged(e,t){this.hasTablesSelectionFormOutlet&&(t.id!==e.id&&this.tablesSelectionFormOutlet.visible(t.id,!1),this.tablesSelectionFormOutlet.visible(e.id,!0),this.checkedValue=this.tablesSelectionFormOutlet.isSelected(e.id),this.update())}checkedValueChanged(){this.hasTablesSelectionFormOutlet&&(this.checkedValue=this.tablesSelectionFormOutlet.isSelected(this.id),this.update())}async update(){return this.updating||=Promise.resolve().then((()=>{this.#t(),delete this.updating})),this.updating}#t(){this.element.querySelector("input").checked=this.checkedValue,this.dispatch("select",{detail:{id:this.id,selected:this.checkedValue}})}}class i{constructor(){this.tokens=[],this.values=null}parse(e){const t=new h(e);for(;!t.isEos();){this.push(this.skipWhitespace(t));const e=this.takeTagged(t)||this.takeUntagged(t);if(!this.push(e))break}return this}push(e){return e&&(this.values?this.values.push(e):this.tokens.push(e)),!!e}skipWhitespace(e){if(e.scan(/\s+/))return new n(e.matched())}takeUntagged(e){if(e.scan(/\S+/))return new l(e.matched())}takeTagged(e){if(!e.scan(/(\w+(?:\.\w+)?)(:\s*)/))return;const t=e.valueAt(1),s=e.valueAt(2),i=this.takeArrayValue(e)||this.takeSingleValue(e)||new n;return new a(t,s,i)}takeArrayValue(e){if(!e.scan(/\[\s*/))return;const t=new n(e.matched()),s=this.values=[];for(;!e.isEos()&&this.push(this.takeSingleValue(e))&&this.push(this.takeDelimiter(e)););e.scan(/\s*]/);const i=new n(e.matched());return this.values=null,new o(t,s,i)}takeDelimiter(e){if(e.scan(/\s*,\s*/))return new n(e.matched())}takeSingleValue(e){return this.takeQuotedValue(e)||this.takeUnquotedValue(e)}takeQuotedValue(e){if(e.scan(/"([^"]*)"/))return new r(e.matched())}takeUnquotedValue(e){if(e.scan(/[^ \],]*/))return new r(e.matched())}}class n{constructor(e=""){this.value=e}render(){return document.createTextNode(this.value)}}class r extends n{render(){const e=document.createElement("span");return e.className="value",e.innerText=this.value,e}}class a extends n{constructor(e,t,s){super(),this.key=e,this.separator=t,this.value=s}render(){const e=document.createElement("span");e.className="tag";const t=document.createElement("span");return t.className="key",t.innerText=this.key,e.appendChild(t),e.appendChild(document.createTextNode(this.separator)),e.appendChild(this.value.render()),e}}class l extends n{render(){const e=document.createElement("span");return e.className="untagged",e.innerText=this.value,e}}let o=class extends n{constructor(e,t,s){super(),this.start=e,this.values=t,this.end=s}render(){const e=document.createElement("span");return e.className="array-values",e.appendChild(this.start.render()),this.values.forEach((t=>{const s=document.createElement("span");s.appendChild(t.render()),e.appendChild(s)})),e.appendChild(this.end.render()),e}};class h{constructor(e){this.input=e,this.position=0,this.last=null}isEos(){return this.position>=this.input.length}scan(e){const t=e.exec(this.input.substring(this.position));return 0===t?.index?(this.last=t,this.position+=t[0].length,!0):(this.last={},!1)}matched(){return this.last&&this.last[0]}valueAt(e){return this.last&&this.last[e]}}const u=[{identifier:"tables--orderable--item",controllerConstructor:class extends e{static values={params:Object};connect(){var e;this.index=(e=this.row,Array.from(e.parentElement.children).indexOf(e))}paramsValueChanged(e){this.id=e.id_value}dragUpdate(e){this.dragOffset=e,this.row.style.position="relative",this.row.style.top=e+"px",this.row.style.zIndex="1",this.row.toggleAttribute("dragging",!0)}updateVisually(e){this.row.style.position="relative",this.row.style.top=this.row.offsetHeight*(e-this.dragIndex)+"px"}updateIndex(e){this.index=e}params(e){const{id_name:t,id_value:s,index_name:i}=this.paramsValue;return[{name:`${e}[${s}][${t}]`,value:this.id},{name:`${e}[${s}][${i}]`,value:this.index}]}reset(){delete this.dragOffset,this.row.removeAttribute("style"),this.row.removeAttribute("dragging")}get hasChanges(){return this.paramsValue.index_value!==this.index}get dragIndex(){return this.dragOffset&&0!==this.dragOffset?this.index+Math.round(this.dragOffset/this.row.offsetHeight):this.index}get comparisonIndex(){return this.dragOffset?this.dragIndex+(this.dragOffset>0?.5:-.5):this.index}get row(){return this.element.parentElement}}},{identifier:"tables--orderable--list",controllerConstructor:class extends e{static outlets=["tables--orderable--item","tables--orderable--form"];startDragging(e){this.dragState=e,document.addEventListener("mousemove",this.mousemove),document.addEventListener("mouseup",this.mouseup),window.addEventListener("scroll",this.scroll,!0),this.element.style.position="relative"}stopDragging(){const e=this.dragState;return delete this.dragState,document.removeEventListener("mousemove",this.mousemove),document.removeEventListener("mouseup",this.mouseup),window.removeEventListener("scroll",this.scroll,!0),this.element.removeAttribute("style"),this.tablesOrderableItemOutlets.forEach((e=>e.reset())),e}drop(){const e=this.dragItem;if(!e)return;const t=e.dragIndex,s=this.tablesOrderableItemOutlets[t];s&&(t<e.index?s.row.insertAdjacentElement("beforebegin",e.row):t>e.index&&s.row.insertAdjacentElement("afterend",e.row),this.tablesOrderableItemOutlets.forEach(((e,t)=>e.updateIndex(t))),this.commitChanges())}commitChanges(){this.tablesOrderableFormOutlet.clear(),this.tablesOrderableItemOutlets.forEach((e=>{e.hasChanges&&this.tablesOrderableFormOutlet.add(e)})),this.tablesOrderableFormOutlet.submit()}mousedown(e){if(this.isDragging)return;const s=this.#s(e.target);s&&(e.preventDefault(),this.startDragging(new t(this.element,e,s.id)),this.dragState.updateCursor(this.element,s.row,e,this.animate))}mousemove=e=>{this.isDragging&&(e.preventDefault(),this.ticking||(this.ticking=!0,window.requestAnimationFrame((()=>{this.ticking=!1,this.dragState.updateCursor(this.element,this.dragItem.row,e,this.animate)}))))};scroll=e=>{this.isDragging&&!this.ticking&&(this.ticking=!0,window.requestAnimationFrame((()=>{this.ticking=!1,this.dragState.updateScroll(this.element,this.dragItem.row,this.animate)})))};mouseup=e=>{this.isDragging&&(this.drop(),this.stopDragging(),this.tablesOrderableFormOutlets.forEach((e=>delete e.dragState)))};tablesOrderableFormOutletConnected(e,t){e.dragState&&this.startDragging(e.dragState)}tablesOrderableFormOutletDisconnected(e,t){this.isDragging&&(e.dragState=this.stopDragging())}animate=e=>{const t=this.dragItem;t.dragUpdate(e),this.#i.forEach(((e,s)=>{e!==t&&e.updateVisually(s)}))};get isDragging(){return!!this.dragState}get dragItem(){return this.isDragging?this.tablesOrderableItemOutlets.find((e=>e.id===this.dragState.targetId)):null}get#i(){return this.tablesOrderableItemOutlets.toSorted(((e,t)=>e.comparisonIndex-t.comparisonIndex))}#s(e){return this.tablesOrderableItemOutlets.find((t=>t.element===e))}}},{identifier:"tables--orderable--form",controllerConstructor:class extends e{static values={scope:String};add(e){e.params(this.scopeValue).forEach((({name:e,value:t})=>{this.element.insertAdjacentHTML("beforeend",`<input type="hidden" name="${e}" value="${t}" data-generated>`)}))}submit(){0!==this.inputs.length&&this.element.requestSubmit()}clear(){this.inputs.forEach((e=>e.remove()))}get inputs(){return this.element.querySelectorAll("input[data-generated]")}}},{identifier:"tables--selection--form",controllerConstructor:class extends e{static values={count:Number,primaryKey:{type:String,default:"id"}};static targets=["count","singular","plural"];connect(){this.countValue=this.inputs.length}toggle(e){const t=this.input(e);return t?t.remove():this.element.insertAdjacentHTML("beforeend",`<input type="hidden" name="${this.primaryKeyValue}[]" value="${e}">`),this.countValue=this.visibleInputs.length,!t}visible(e,t){const s=this.input(e);return s&&(s.disabled=!t),this.countValue=this.visibleInputs.length,!s}isSelected(e){return!!this.input(e)}get inputs(){return this.element.querySelectorAll(`input[name="${this.primaryKeyValue}[]"]`)}get visibleInputs(){return Array.from(this.inputs).filter((e=>!e.disabled))}input(e){return this.element.querySelector(`input[name="${this.primaryKeyValue}[]"][value="${e}"]`)}countValueChanged(e){this.element.toggleAttribute("hidden",0===e),this.countTarget.textContent=e,this.singularTarget.toggleAttribute("hidden",1!==e),this.pluralTarget.toggleAttribute("hidden",1===e)}}},{identifier:"tables--selection--item",controllerConstructor:s},{identifier:"tables--selection--table",controllerConstructor:class extends e{static targets=["header","item"];static outlets=["tables--selection--form"];itemTargetConnected(e){this.update()}itemTargetDisconnected(e){this.update()}toggleHeader(e){this.items.forEach((t=>{t.checkedValue!==e.target.checked&&(t.checkedValue=this.tablesSelectionFormOutlet.toggle(t.id))}))}async update(){return this.updating||=Promise.resolve().then((()=>{this.#t(),delete this.updating})),this.updating}#t(){let e=0,t=0;this.items.forEach((s=>{e++,s.checkedValue&&t++})),this.headerInput.checked=e>0&&t===e,this.headerInput.indeterminate=t>0&&t!==e}get headerInput(){return this.headerTarget.querySelector("input")}get items(){return this.itemTargets.map((e=>this.#n(e))).filter((e=>e))}#n(e){return this.application.getControllerForElementAndIdentifier(e,"tables--selection--item")}}},{identifier:"tables--query",controllerConstructor:class extends e{static targets=["modal"];disconnect(){delete this.pending,document.removeEventListener("selectionchange",this.selection)}focus(){document.activeElement!==this.query&&(this.query.addEventListener("focusin",(e=>{e.target.setSelectionRange(-1,-1)}),{once:!0}),this.query.focus())}closeModal(){delete this.modalTarget.dataset.open,document.activeElement===this.query&&document.activeElement.blur(),document.removeEventListener("selectionchange",this.selection)}openModal(){this.modalTarget.dataset.open=!0,document.addEventListener("selectionchange",this.selection)}clear(){""===this.query.value?this.closeModal():(this.query.value="",this.query.dispatchEvent(new Event("input")),this.query.dispatchEvent(new Event("change")),this.update())}submit(){const e=this.isFocused,t=e&&this.query.selectionStart;this.pending&&(clearTimeout(this.pending),delete this.pending),""===this.query.value&&(this.query.disabled=!0,setTimeout((()=>{this.query.disabled=!1,e&&this.query.focus()}),0)),e&&t?(this.position.value=t,this.position.disabled=!1):(this.position.value="",this.position.disabled=!0)}update=()=>{this.pending&&clearTimeout(this.pending),this.pending=setTimeout((()=>{this.element.requestSubmit()}),300)};selection=()=>{this.isFocused&&this.update()};beforeMorphAttribute(e){if("data-open"===e.detail.attributeName)e.preventDefault()}get query(){return this.element.querySelector("[role=searchbox]")}get position(){return this.element.querySelector("input[name=p]")}get isFocused(){return this.query===document.activeElement}}},{identifier:"tables--query-input",controllerConstructor:class extends e{static targets=["input","highlight"];static values={query:String};connect(){this.queryValue=this.inputTarget.value,this.element.dataset.connected=""}disconnect(){delete this.element.dataset.connected}update(){this.queryValue=this.inputTarget.value}queryValueChanged(e){this.highlightTarget.innerHTML="",(new i).parse(e).tokens.forEach((e=>{this.highlightTarget.appendChild(e.render())}))}}}];export{u as default};
|
1
|
+
import{Controller as e}from"@hotwired/stimulus";class t{constructor(e,t,s){this.cursorOffset=t.offsetY,this.initialPosition=t.target.offsetTop-e.offsetTop,this.targetId=s}updateCursor(e,t,s,i){this.listOffset=e.getBoundingClientRect().top;let n=s.clientY-this.listOffset-this.cursorOffset;this.#e(e,t,n,i)}updateScroll(e,t,s){const i=this.listOffset;this.listOffset=e.getBoundingClientRect().top;const n=i-this.listOffset,r=this.position+n;this.#e(e,t,r,s)}#e(e,t,s,i){s=Math.max(s,0),s=Math.min(s,e.offsetHeight-t.offsetHeight),this.position=s;i(s-this.initialPosition)}}class s extends e{static outlets=["tables--selection--form"];static values={params:Object,checked:Boolean};tablesSelectionFormOutletConnected(e){e.visible(this.id,!0),this.checkedValue=e.isSelected(this.id)}disconnect(){this.hasTablesSelectionFormOutlet&&this.tablesSelectionFormOutlet.visible(this.id,!1)}change(e){e.preventDefault(),this.checkedValue=this.tablesSelectionFormOutlet.toggle(this.id)}get id(){return this.paramsValue.id}paramsValueChanged(e,t){this.hasTablesSelectionFormOutlet&&(t.id!==e.id&&this.tablesSelectionFormOutlet.visible(t.id,!1),this.tablesSelectionFormOutlet.visible(e.id,!0),this.checkedValue=this.tablesSelectionFormOutlet.isSelected(e.id),this.update())}checkedValueChanged(){this.hasTablesSelectionFormOutlet&&(this.checkedValue=this.tablesSelectionFormOutlet.isSelected(this.id),this.update())}async update(){return this.updating||=Promise.resolve().then((()=>{this.#t(),delete this.updating})),this.updating}#t(){this.element.querySelector("input").checked=this.checkedValue,this.dispatch("select",{detail:{id:this.id,selected:this.checkedValue}})}}class i{constructor(){this.tokens=[],this.values=null}parse(e){const t=new h(e);for(;!t.isEos();){this.push(this.skipWhitespace(t));const e=this.takeTagged(t)||this.takeUntagged(t);if(!this.push(e))break}return this}push(e){return e&&(this.values?this.values.push(e):this.tokens.push(e)),!!e}skipWhitespace(e){if(e.scan(/\s+/))return new n(e.matched())}takeUntagged(e){if(e.scan(/\S+/))return new l(e.matched())}takeTagged(e){if(!e.scan(/(\w+(?:\.\w+)?)(:\s*)/))return;const t=e.valueAt(1),s=e.valueAt(2),i=this.takeArrayValue(e)||this.takeSingleValue(e)||new n;return new a(t,s,i)}takeArrayValue(e){if(!e.scan(/\[\s*/))return;const t=new n(e.matched()),s=this.values=[];for(;!e.isEos()&&this.push(this.takeSingleValue(e))&&this.push(this.takeDelimiter(e)););e.scan(/\s*]/);const i=new n(e.matched());return this.values=null,new o(t,s,i)}takeDelimiter(e){if(e.scan(/\s*,\s*/))return new n(e.matched())}takeSingleValue(e){return this.takeQuotedValue(e)||this.takeUnquotedValue(e)}takeQuotedValue(e){if(e.scan(/"([^"]*)"/))return new r(e.matched())}takeUnquotedValue(e){if(e.scan(/[^ \],]*/))return new r(e.matched())}}class n{constructor(e=""){this.value=e}render(){return document.createTextNode(this.value)}}class r extends n{render(){const e=document.createElement("span");return e.className="value",e.innerText=this.value,e}}class a extends n{constructor(e,t,s){super(),this.key=e,this.separator=t,this.value=s}render(){const e=document.createElement("span");e.className="tag";const t=document.createElement("span");return t.className="key",t.innerText=this.key,e.appendChild(t),e.appendChild(document.createTextNode(this.separator)),e.appendChild(this.value.render()),e}}class l extends n{render(){const e=document.createElement("span");return e.className="untagged",e.innerText=this.value,e}}let o=class extends n{constructor(e,t,s){super(),this.start=e,this.values=t,this.end=s}render(){const e=document.createElement("span");return e.className="array-values",e.appendChild(this.start.render()),this.values.forEach((t=>{const s=document.createElement("span");s.appendChild(t.render()),e.appendChild(s)})),e.appendChild(this.end.render()),e}};class h{constructor(e){this.input=e,this.position=0,this.last=null}isEos(){return this.position>=this.input.length}scan(e){const t=e.exec(this.input.substring(this.position));return 0===t?.index?(this.last=t,this.position+=t[0].length,!0):(this.last={},!1)}matched(){return this.last&&this.last[0]}valueAt(e){return this.last&&this.last[e]}}const u=[{identifier:"tables--orderable--item",controllerConstructor:class extends e{static values={params:Object};connect(){var e;this.index=(e=this.row,Array.from(e.parentElement.children).indexOf(e))}paramsValueChanged(e){this.id=e.id_value}dragUpdate(e){this.dragOffset=e,this.row.style.position="relative",this.row.style.top=e+"px",this.row.style.zIndex="1",this.row.toggleAttribute("dragging",!0)}updateVisually(e){this.row.style.position="relative",this.row.style.top=this.row.offsetHeight*(e-this.dragIndex)+"px"}updateIndex(e){this.index=e}params(e){const{id_name:t,id_value:s,index_name:i}=this.paramsValue;return[{name:`${e}[${s}][${t}]`,value:this.id},{name:`${e}[${s}][${i}]`,value:this.index}]}reset(){delete this.dragOffset,this.row.removeAttribute("style"),this.row.removeAttribute("dragging")}get hasChanges(){return this.paramsValue.index_value!==this.index}get dragIndex(){return this.dragOffset&&0!==this.dragOffset?this.index+Math.round(this.dragOffset/this.row.offsetHeight):this.index}get comparisonIndex(){return this.dragOffset?this.dragIndex+(this.dragOffset>0?.5:-.5):this.index}get row(){return this.element.parentElement}}},{identifier:"tables--orderable--list",controllerConstructor:class extends e{static outlets=["tables--orderable--item","tables--orderable--form"];startDragging(e){this.dragState=e,document.addEventListener("mousemove",this.mousemove),document.addEventListener("mouseup",this.mouseup),window.addEventListener("scroll",this.scroll,!0),this.element.style.position="relative"}stopDragging(){const e=this.dragState;return delete this.dragState,document.removeEventListener("mousemove",this.mousemove),document.removeEventListener("mouseup",this.mouseup),window.removeEventListener("scroll",this.scroll,!0),this.element.removeAttribute("style"),this.tablesOrderableItemOutlets.forEach((e=>e.reset())),e}drop(){const e=this.dragItem;if(!e)return;const t=e.dragIndex,s=this.tablesOrderableItemOutlets[t];s&&(t<e.index?s.row.insertAdjacentElement("beforebegin",e.row):t>e.index&&s.row.insertAdjacentElement("afterend",e.row),this.tablesOrderableItemOutlets.forEach(((e,t)=>e.updateIndex(t))),this.commitChanges())}commitChanges(){this.tablesOrderableFormOutlet.clear(),this.tablesOrderableItemOutlets.forEach((e=>{e.hasChanges&&this.tablesOrderableFormOutlet.add(e)})),this.tablesOrderableFormOutlet.submit()}mousedown(e){if(this.isDragging)return;const s=this.#s(e.target);s&&(e.preventDefault(),this.startDragging(new t(this.element,e,s.id)),this.dragState.updateCursor(this.element,s.row,e,this.animate))}mousemove=e=>{this.isDragging&&(e.preventDefault(),this.ticking||(this.ticking=!0,window.requestAnimationFrame((()=>{this.ticking=!1,this.dragState.updateCursor(this.element,this.dragItem.row,e,this.animate)}))))};scroll=e=>{this.isDragging&&!this.ticking&&(this.ticking=!0,window.requestAnimationFrame((()=>{this.ticking=!1,this.dragState.updateScroll(this.element,this.dragItem.row,this.animate)})))};mouseup=e=>{this.isDragging&&(this.drop(),this.stopDragging(),this.tablesOrderableFormOutlets.forEach((e=>delete e.dragState)))};tablesOrderableFormOutletConnected(e,t){e.dragState&&this.startDragging(e.dragState)}tablesOrderableFormOutletDisconnected(e,t){this.isDragging&&(e.dragState=this.stopDragging())}animate=e=>{const t=this.dragItem;t.dragUpdate(e),this.#i.forEach(((e,s)=>{e!==t&&e.updateVisually(s)}))};get isDragging(){return!!this.dragState}get dragItem(){return this.isDragging?this.tablesOrderableItemOutlets.find((e=>e.id===this.dragState.targetId)):null}get#i(){return this.tablesOrderableItemOutlets.toSorted(((e,t)=>e.comparisonIndex-t.comparisonIndex))}#s(e){return this.tablesOrderableItemOutlets.find((t=>t.element===e))}}},{identifier:"tables--orderable--form",controllerConstructor:class extends e{static values={scope:String};add(e){e.params(this.scopeValue).forEach((({name:e,value:t})=>{this.element.insertAdjacentHTML("beforeend",`<input type="hidden" name="${e}" value="${t}" data-generated>`)}))}submit(){0!==this.inputs.length&&this.element.requestSubmit()}clear(){this.inputs.forEach((e=>e.remove()))}get inputs(){return this.element.querySelectorAll("input[data-generated]")}}},{identifier:"tables--selection--form",controllerConstructor:class extends e{static values={count:Number,primaryKey:{type:String,default:"id"}};static targets=["count","singular","plural"];connect(){this.countValue=this.inputs.length}toggle(e){const t=this.input(e);return t?t.remove():this.element.insertAdjacentHTML("beforeend",`<input type="hidden" name="${this.primaryKeyValue}[]" value="${e}">`),this.countValue=this.visibleInputs.length,!t}visible(e,t){const s=this.input(e);return s&&(s.disabled=!t),this.countValue=this.visibleInputs.length,!s}isSelected(e){return!!this.input(e)}get inputs(){return this.element.querySelectorAll(`input[name="${this.primaryKeyValue}[]"]`)}get visibleInputs(){return Array.from(this.inputs).filter((e=>!e.disabled))}input(e){return this.element.querySelector(`input[name="${this.primaryKeyValue}[]"][value="${e}"]`)}countValueChanged(e){this.element.toggleAttribute("hidden",0===e),this.countTarget.textContent=e,this.singularTarget.toggleAttribute("hidden",1!==e),this.pluralTarget.toggleAttribute("hidden",1===e)}}},{identifier:"tables--selection--item",controllerConstructor:s},{identifier:"tables--selection--table",controllerConstructor:class extends e{static targets=["header","item"];static outlets=["tables--selection--form"];itemTargetConnected(e){this.update()}itemTargetDisconnected(e){this.update()}toggleHeader(e){this.items.forEach((t=>{t.checkedValue!==e.target.checked&&(t.checkedValue=this.tablesSelectionFormOutlet.toggle(t.id))}))}async update(){return this.updating||=Promise.resolve().then((()=>{this.#t(),delete this.updating})),this.updating}#t(){let e=0,t=0;this.items.forEach((s=>{e++,s.checkedValue&&t++})),this.headerInput.checked=e>0&&t===e,this.headerInput.indeterminate=t>0&&t!==e}get headerInput(){return this.headerTarget.querySelector("input")}get items(){return this.itemTargets.map((e=>this.#n(e))).filter((e=>e))}#n(e){return this.application.getControllerForElementAndIdentifier(e,"tables--selection--item")}}},{identifier:"tables--query",controllerConstructor:class extends e{static targets=["modal"];disconnect(){delete this.pending,document.removeEventListener("selectionchange",this.selection)}focus(){document.activeElement!==this.query&&(this.query.addEventListener("focusin",(e=>{e.target.setSelectionRange(-1,-1)}),{once:!0}),this.query.focus())}closeModal(){delete this.modalTarget.dataset.open,document.activeElement===this.query&&document.activeElement.blur(),document.removeEventListener("selectionchange",this.selection)}openModal(){this.modalTarget.dataset.open=!0,document.addEventListener("selectionchange",this.selection)}clear(){""===this.query.value?this.closeModal():(this.query.value="",this.query.dispatchEvent(new Event("input")),this.query.dispatchEvent(new Event("change")),this.update())}submit(){const e=this.isFocused,t=e&&this.query.selectionStart;this.pending&&(clearTimeout(this.pending),delete this.pending),""===this.query.value&&(this.query.disabled=!0,setTimeout((()=>{this.query.disabled=!1,e&&this.query.focus()}),0)),e&&t?(this.position.value=t,this.position.disabled=!1):(this.position.value="",this.position.disabled=!0)}update=()=>{this.pending&&clearTimeout(this.pending),this.pending=setTimeout((()=>{this.element.requestSubmit()}),300)};selection=()=>{this.isFocused&&this.query.value.length>0&&this.update()};beforeMorphAttribute(e){if("data-open"===e.detail.attributeName)e.preventDefault()}get query(){return this.element.querySelector("[role=searchbox]")}get position(){return this.element.querySelector("input[name=p]")}get isFocused(){return this.query===document.activeElement}}},{identifier:"tables--query-input",controllerConstructor:class extends e{static targets=["input","highlight"];static values={query:String};connect(){this.queryValue=this.inputTarget.value,this.element.dataset.connected=""}disconnect(){delete this.element.dataset.connected}update(){this.queryValue=this.inputTarget.value}queryValueChanged(e){this.highlightTarget.innerHTML="",(new i).parse(e).tokens.forEach((e=>{this.highlightTarget.appendChild(e.render())}))}}}];export{u as default};
|
2
2
|
//# sourceMappingURL=tables.min.js.map
|
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"tables.min.js","sources":["../../../javascript/tables/orderable/list_controller.js","../../../javascript/tables/selection/item_controller.js","../../../javascript/tables/query_input_controller.js","../../../javascript/tables/application.js","../../../javascript/tables/orderable/item_controller.js","../../../javascript/tables/orderable/form_controller.js","../../../javascript/tables/selection/form_controller.js","../../../javascript/tables/selection/table_controller.js","../../../javascript/tables/query_controller.js"],"sourcesContent":["import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableListController extends Controller {\n static outlets = [\"tables--orderable--item\", \"tables--orderable--form\"];\n\n //region State transitions\n\n startDragging(dragState) {\n this.dragState = dragState;\n\n document.addEventListener(\"mousemove\", this.mousemove);\n document.addEventListener(\"mouseup\", this.mouseup);\n window.addEventListener(\"scroll\", this.scroll, true);\n\n this.element.style.position = \"relative\";\n }\n\n stopDragging() {\n const dragState = this.dragState;\n delete this.dragState;\n\n document.removeEventListener(\"mousemove\", this.mousemove);\n document.removeEventListener(\"mouseup\", this.mouseup);\n window.removeEventListener(\"scroll\", this.scroll, true);\n\n this.element.removeAttribute(\"style\");\n this.tablesOrderableItemOutlets.forEach((item) => item.reset());\n\n return dragState;\n }\n\n drop() {\n // note: early returns guard against turbo updates that prevent us finding\n // the right item to drop on. In this situation it's better to discard the\n // drop than to drop in the wrong place.\n\n const dragItem = this.dragItem;\n\n if (!dragItem) return;\n\n const newIndex = dragItem.dragIndex;\n const targetItem = this.tablesOrderableItemOutlets[newIndex];\n\n if (!targetItem) return;\n\n // swap the dragged item into the correct position for its current offset\n if (newIndex < dragItem.index) {\n targetItem.row.insertAdjacentElement(\"beforebegin\", dragItem.row);\n } else if (newIndex > dragItem.index) {\n targetItem.row.insertAdjacentElement(\"afterend\", dragItem.row);\n }\n\n // reindex all items based on their new positions\n this.tablesOrderableItemOutlets.forEach((item, index) =>\n item.updateIndex(index),\n );\n\n // save the changes\n this.commitChanges();\n }\n\n commitChanges() {\n // clear any existing inputs to prevent duplicates\n this.tablesOrderableFormOutlet.clear();\n\n // insert any items that have changed position\n this.tablesOrderableItemOutlets.forEach((item) => {\n if (item.hasChanges) this.tablesOrderableFormOutlet.add(item);\n });\n\n this.tablesOrderableFormOutlet.submit();\n }\n\n //endregion\n\n //region Events\n\n mousedown(event) {\n if (this.isDragging) return;\n\n const target = this.#targetItem(event.target);\n\n if (!target) return;\n\n event.preventDefault(); // prevent built-in drag\n\n this.startDragging(new DragState(this.element, event, target.id));\n\n this.dragState.updateCursor(this.element, target.row, event, this.animate);\n }\n\n mousemove = (event) => {\n if (!this.isDragging) return;\n\n event.preventDefault(); // prevent build-in drag\n\n if (this.ticking) return;\n\n this.ticking = true;\n\n window.requestAnimationFrame(() => {\n this.ticking = false;\n this.dragState.updateCursor(\n this.element,\n this.dragItem.row,\n event,\n this.animate,\n );\n });\n };\n\n scroll = (event) => {\n if (!this.isDragging || this.ticking) return;\n\n this.ticking = true;\n\n window.requestAnimationFrame(() => {\n this.ticking = false;\n this.dragState.updateScroll(\n this.element,\n this.dragItem.row,\n this.animate,\n );\n });\n };\n\n mouseup = (event) => {\n if (!this.isDragging) return;\n\n this.drop();\n this.stopDragging();\n this.tablesOrderableFormOutlets.forEach((form) => delete form.dragState);\n };\n\n tablesOrderableFormOutletConnected(form, element) {\n if (form.dragState) {\n // restore the previous controller's state\n this.startDragging(form.dragState);\n }\n }\n\n tablesOrderableFormOutletDisconnected(form, element) {\n if (this.isDragging) {\n // cache drag state in the form\n form.dragState = this.stopDragging();\n }\n }\n\n //endregion\n\n //region Helpers\n\n /**\n * Updates the position of the drag item with a relative offset. Updates\n * other items relative to the new position of the drag item, as required.\n *\n * @callback {OrderableListController~animate}\n * @param {number} offset\n */\n animate = (offset) => {\n const dragItem = this.dragItem;\n\n // Visually update the dragItem so it follows the cursor\n dragItem.dragUpdate(offset);\n\n // Visually updates the position of all items in the list relative to the\n // dragged item. No actual changes to orderings at this stage.\n this.#currentItems.forEach((item, index) => {\n if (item === dragItem) return;\n item.updateVisually(index);\n });\n };\n\n get isDragging() {\n return !!this.dragState;\n }\n\n get dragItem() {\n if (!this.isDragging) return null;\n\n return this.tablesOrderableItemOutlets.find(\n (item) => item.id === this.dragState.targetId,\n );\n }\n\n /**\n * Returns the current items in the list, sorted by their current index.\n * Current uses the drag index if the item is being dragged, if set.\n *\n * @returns {Array[OrderableRowController]}\n */\n get #currentItems() {\n return this.tablesOrderableItemOutlets.toSorted(\n (a, b) => a.comparisonIndex - b.comparisonIndex,\n );\n }\n\n /**\n * Returns the item outlet that was clicked on, if any.\n *\n * @param element {HTMLElement} the clicked ordinal cell\n * @returns {OrderableRowController}\n */\n #targetItem(element) {\n return this.tablesOrderableItemOutlets.find(\n (item) => item.element === element,\n );\n }\n\n //endregion\n}\n\n/**\n * During drag we want to be able to translate a document-relative coordinate\n * into a coordinate relative to the list element. This state object calculates\n * and stores internal state so that we can translate absolute page coordinates\n * from mouse events into relative offsets for the list items within the list\n * element.\n *\n * We also keep track of the drag target so that if the controller is attached\n * to a new element during the drag we can continue after the turbo update.\n */\nclass DragState {\n /**\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param event {MouseEvent} the initial event\n * @param id {String} the id of the element being dragged\n */\n constructor(list, event, id) {\n // cursor offset is the offset of the cursor relative to the drag item\n this.cursorOffset = event.offsetY;\n\n // initial offset is the offset position of the drag item at drag start\n this.initialPosition = event.target.offsetTop - list.offsetTop;\n\n // id of the item being dragged\n this.targetId = id;\n }\n\n /**\n * Calculates the offset of the drag item relative to its initial position.\n *\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param row {HTMLElement} the row being dragged\n * @param event {MouseEvent} the current event\n * @param callback {OrderableListController~animate} updates the drag item with a relative offset\n */\n updateCursor(list, row, event, callback) {\n // Calculate and store the list offset relative to the viewport\n // This value is cached so we can calculate the outcome of any scroll events\n this.listOffset = list.getBoundingClientRect().top;\n\n // Calculate the position of the cursor relative to the list.\n // Accounts for scroll offsets by using the item's bounding client rect.\n const cursorPosition = event.clientY - this.listOffset;\n\n // intended item position relative to the list, from cursor position\n let itemPosition = cursorPosition - this.cursorOffset;\n\n this.#updateItemPosition(list, row, itemPosition, callback);\n }\n\n /**\n * Animates the item's position as the list scrolls. Requires a previous call\n * to set the scroll offset.\n *\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param row {HTMLElement} the row being dragged\n * @param callback {OrderableListController~animate} updates the drag item with a relative offset\n */\n updateScroll(list, row, callback) {\n const previousScrollOffset = this.listOffset;\n\n // Calculate and store the list offset relative to the viewport\n // This value is cached so we can calculate the outcome of any scroll events\n this.listOffset = list.getBoundingClientRect().top;\n\n // Calculate the change in scroll offset since the last update\n const scrollDelta = previousScrollOffset - this.listOffset;\n\n // intended item position relative to the list, from cursor position\n const position = this.position + scrollDelta;\n\n this.#updateItemPosition(list, row, position, callback);\n }\n\n #updateItemPosition(list, row, position, callback) {\n // ensure itemPosition is within the bounds of the list (tbody)\n position = Math.max(position, 0);\n position = Math.min(position, list.offsetHeight - row.offsetHeight);\n\n // cache the item's position relative to the list for use in scroll events\n this.position = position;\n\n // Item has position: relative, so we want to calculate the amount to move\n // the item relative to it's DOM position to represent how much it has been\n // dragged by.\n const offset = position - this.initialPosition;\n\n // Convert itemPosition from offset relative to list to offset relative to\n // its position within the DOM (if it hadn't moved).\n callback(offset);\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\n/**\n * Couples an input element in a row to the selection form which is turbo-permanent and outside the table.\n * When the input is toggled, the form will create/destroy hidden inputs. The checkbox inside this cell will follow\n * the hidden inputs.\n *\n * Cell value may change when:\n * * cell connects, e.g. when the user paginates\n * * cell is re-used by turbo-morph, e.g. pagination\n * * cell is toggled\n * * select-all/de-select-all on table header\n */\nexport default class SelectionItemController extends Controller {\n static outlets = [\"tables--selection--form\"];\n static values = {\n params: Object,\n checked: Boolean,\n };\n\n tablesSelectionFormOutletConnected(form) {\n form.visible(this.id, true);\n this.checkedValue = form.isSelected(this.id);\n }\n\n disconnect() {\n // Remove from form's list of visible selections.\n // This should be an outlet disconnect, but those events are not reliable in turbo 8.0\n if (this.hasTablesSelectionFormOutlet) {\n this.tablesSelectionFormOutlet.visible(this.id, false);\n }\n }\n\n change(e) {\n e.preventDefault();\n\n this.checkedValue = this.tablesSelectionFormOutlet.toggle(this.id);\n }\n\n get id() {\n return this.paramsValue.id;\n }\n\n /**\n * Update checked to match match selection form. This occurs when the item is re-used by turbo-morph.\n */\n paramsValueChanged(params, previous) {\n if (!this.hasTablesSelectionFormOutlet) return;\n\n // if id is changing (e.g. morph) then let the form know that the previous id is now not visible\n if (previous.id !== params.id) {\n this.tablesSelectionFormOutlet.visible(previous.id, false);\n }\n\n // tell form that our id is now visible in the table\n this.tablesSelectionFormOutlet.visible(params.id, true);\n\n // id has changed, so update checked from form\n this.checkedValue = this.tablesSelectionFormOutlet.isSelected(params.id);\n\n // propagate changes\n this.update();\n }\n\n /**\n * Update input to match checked. This occurs when the item is toggled, connected, or morphed.\n */\n checkedValueChanged() {\n if (!this.hasTablesSelectionFormOutlet) return;\n\n // ensure that checked matches the form, i.e. if morphed\n this.checkedValue = this.tablesSelectionFormOutlet.isSelected(this.id);\n\n // propagate changes\n this.update();\n }\n\n /**\n * Notify table that id or value may have changed. Note that this may fire when nothing has changed.\n *\n * Debouncing to minimise dom updates.\n */\n async update() {\n this.updating ||= Promise.resolve().then(() => {\n this.#update();\n delete this.updating;\n });\n\n return this.updating;\n }\n\n #update() {\n this.element.querySelector(\"input\").checked = this.checkedValue;\n this.dispatch(\"select\", {\n detail: { id: this.id, selected: this.checkedValue },\n });\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class QueryInputController extends Controller {\n static targets = [\"input\", \"highlight\"];\n static values = { query: String };\n\n connect() {\n this.queryValue = this.inputTarget.value;\n this.element.dataset.connected = \"\";\n }\n\n disconnect() {\n delete this.element.dataset.connected;\n }\n\n update() {\n this.queryValue = this.inputTarget.value;\n }\n\n queryValueChanged(query) {\n this.highlightTarget.innerHTML = \"\";\n\n new Parser().parse(query).tokens.forEach((token) => {\n this.highlightTarget.appendChild(token.render());\n });\n }\n}\n\nclass Parser {\n constructor() {\n this.tokens = [];\n this.values = null;\n }\n\n parse(input) {\n const query = new StringScanner(input);\n\n while (!query.isEos()) {\n this.push(this.skipWhitespace(query));\n\n const value = this.takeTagged(query) || this.takeUntagged(query);\n\n if (!this.push(value)) break;\n }\n\n return this;\n }\n\n push(token) {\n if (token) {\n this.values ? this.values.push(token) : this.tokens.push(token);\n }\n\n return !!token;\n }\n\n skipWhitespace(query) {\n if (!query.scan(/\\s+/)) return;\n\n return new Token(query.matched());\n }\n\n takeUntagged(query) {\n if (!query.scan(/\\S+/)) return;\n\n return new Untagged(query.matched());\n }\n\n takeTagged(query) {\n if (!query.scan(/(\\w+(?:\\.\\w+)?)(:\\s*)/)) return;\n\n const key = query.valueAt(1);\n const separator = query.valueAt(2);\n\n const value =\n this.takeArrayValue(query) || this.takeSingleValue(query) || new Token();\n\n return new Tagged(key, separator, value);\n }\n\n takeArrayValue(query) {\n if (!query.scan(/\\[\\s*/)) return;\n\n const start = new Token(query.matched());\n const values = (this.values = []);\n\n while (!query.isEos()) {\n if (!this.push(this.takeSingleValue(query))) break;\n if (!this.push(this.takeDelimiter(query))) break;\n }\n\n query.scan(/\\s*]/);\n const end = new Token(query.matched());\n\n this.values = null;\n\n return new Array(start, values, end);\n }\n\n takeDelimiter(query) {\n if (!query.scan(/\\s*,\\s*/)) return;\n\n return new Token(query.matched());\n }\n\n takeSingleValue(query) {\n return this.takeQuotedValue(query) || this.takeUnquotedValue(query);\n }\n\n takeQuotedValue(query) {\n if (!query.scan(/\"([^\"]*)\"/)) return;\n\n return new Value(query.matched());\n }\n\n takeUnquotedValue(query) {\n if (!query.scan(/[^ \\],]*/)) return;\n\n return new Value(query.matched());\n }\n}\n\nclass Token {\n constructor(value = \"\") {\n this.value = value;\n }\n\n render() {\n return document.createTextNode(this.value);\n }\n}\n\nclass Value extends Token {\n render() {\n const span = document.createElement(\"span\");\n span.className = \"value\";\n span.innerText = this.value;\n\n return span;\n }\n}\n\nclass Tagged extends Token {\n constructor(key, separator, value) {\n super();\n\n this.key = key;\n this.separator = separator;\n this.value = value;\n }\n\n render() {\n const span = document.createElement(\"span\");\n span.className = \"tag\";\n\n const key = document.createElement(\"span\");\n key.className = \"key\";\n key.innerText = this.key;\n\n span.appendChild(key);\n span.appendChild(document.createTextNode(this.separator));\n span.appendChild(this.value.render());\n\n return span;\n }\n}\n\nclass Untagged extends Token {\n render() {\n const span = document.createElement(\"span\");\n span.className = \"untagged\";\n span.innerText = this.value;\n return span;\n }\n}\n\nclass Array extends Token {\n constructor(start, values, end) {\n super();\n\n this.start = start;\n this.values = values;\n this.end = end;\n }\n\n render() {\n const array = document.createElement(\"span\");\n array.className = \"array-values\";\n array.appendChild(this.start.render());\n\n this.values.forEach((value) => {\n const span = document.createElement(\"span\");\n span.appendChild(value.render());\n array.appendChild(span);\n });\n\n array.appendChild(this.end.render());\n\n return array;\n }\n}\n\nclass StringScanner {\n constructor(input) {\n this.input = input;\n this.position = 0;\n this.last = null;\n }\n\n isEos() {\n return this.position >= this.input.length;\n }\n\n scan(regex) {\n const match = regex.exec(this.input.substring(this.position));\n if (match?.index === 0) {\n this.last = match;\n this.position += match[0].length;\n return true;\n } else {\n this.last = {};\n return false;\n }\n }\n\n matched() {\n return this.last && this.last[0];\n }\n\n valueAt(index) {\n return this.last && this.last[index];\n }\n}\n","import OrderableItemController from \"./orderable/item_controller\";\nimport OrderableListController from \"./orderable/list_controller\";\nimport OrderableFormController from \"./orderable/form_controller\";\nimport SelectionFormController from \"./selection/form_controller\";\nimport SelectionItemController from \"./selection/item_controller\";\nimport SelectionTableController from \"./selection/table_controller\";\nimport QueryController from \"./query_controller\";\nimport QueryInputController from \"./query_input_controller\";\n\nconst Definitions = [\n {\n identifier: \"tables--orderable--item\",\n controllerConstructor: OrderableItemController,\n },\n {\n identifier: \"tables--orderable--list\",\n controllerConstructor: OrderableListController,\n },\n {\n identifier: \"tables--orderable--form\",\n controllerConstructor: OrderableFormController,\n },\n {\n identifier: \"tables--selection--form\",\n controllerConstructor: SelectionFormController,\n },\n {\n identifier: \"tables--selection--item\",\n controllerConstructor: SelectionItemController,\n },\n {\n identifier: \"tables--selection--table\",\n controllerConstructor: SelectionTableController,\n },\n {\n identifier: \"tables--query\",\n controllerConstructor: QueryController,\n },\n {\n identifier: \"tables--query-input\",\n controllerConstructor: QueryInputController,\n },\n];\n\nexport { Definitions as default };\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableRowController extends Controller {\n static values = {\n params: Object,\n };\n\n connect() {\n // index from server may be inconsistent with the visual ordering,\n // especially if this is a new node. Use positional indexes instead,\n // as these are the values we will send on save.\n this.index = domIndex(this.row);\n }\n\n paramsValueChanged(params) {\n this.id = params.id_value;\n }\n\n dragUpdate(offset) {\n this.dragOffset = offset;\n this.row.style.position = \"relative\";\n this.row.style.top = offset + \"px\";\n this.row.style.zIndex = \"1\";\n this.row.toggleAttribute(\"dragging\", true);\n }\n\n /**\n * Called on items that are not the dragged item during drag. Updates the\n * visual position of the item relative to the dragged item.\n *\n * @param index {number} intended index of the item during drag\n */\n updateVisually(index) {\n this.row.style.position = \"relative\";\n this.row.style.top = `${\n this.row.offsetHeight * (index - this.dragIndex)\n }px`;\n }\n\n /**\n * Set the index value of the item. This is called on all items after a drop\n * event. If the index is different to the params index then this item has\n * changed.\n *\n * @param index {number} the new index value\n */\n updateIndex(index) {\n this.index = index;\n }\n\n /** Retrieve params for use in the form */\n params(scope) {\n const { id_name, id_value, index_name } = this.paramsValue;\n return [\n { name: `${scope}[${id_value}][${id_name}]`, value: this.id },\n { name: `${scope}[${id_value}][${index_name}]`, value: this.index },\n ];\n }\n\n /**\n * Restore any visual changes made during drag and remove the drag state.\n */\n reset() {\n delete this.dragOffset;\n this.row.removeAttribute(\"style\");\n this.row.removeAttribute(\"dragging\");\n }\n\n /**\n * @returns {boolean} true when the item has a change to its index value\n */\n get hasChanges() {\n return this.paramsValue.index_value !== this.index;\n }\n\n /**\n * Calculate the relative index of the item during drag. This is used to\n * sort items during drag as it takes into account any uncommitted changes\n * to index caused by the drag offset.\n *\n * @returns {number} index for the purposes of drag and drop ordering\n */\n get dragIndex() {\n if (this.dragOffset && this.dragOffset !== 0) {\n return this.index + Math.round(this.dragOffset / this.row.offsetHeight);\n } else {\n return this.index;\n }\n }\n\n /**\n * Index value for use in comparisons during drag. This is used to determine\n * whether the dragged item is above or below another item. If this item is\n * being dragged then we offset the index by 0.5 to ensure that it jumps up\n * or down when it reaches the midpoint of the item above or below it.\n *\n * @returns {number}\n */\n get comparisonIndex() {\n if (this.dragOffset) {\n return this.dragIndex + (this.dragOffset > 0 ? 0.5 : -0.5);\n } else {\n return this.index;\n }\n }\n\n /**\n * The containing row element.\n *\n * @returns {HTMLElement}\n */\n get row() {\n return this.element.parentElement;\n }\n}\n\nfunction domIndex(element) {\n return Array.from(element.parentElement.children).indexOf(element);\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableFormController extends Controller {\n static values = { scope: String };\n\n add(item) {\n item.params(this.scopeValue).forEach(({ name, value }) => {\n this.element.insertAdjacentHTML(\n \"beforeend\",\n `<input type=\"hidden\" name=\"${name}\" value=\"${value}\" data-generated>`,\n );\n });\n }\n\n submit() {\n if (this.inputs.length === 0) return;\n\n this.element.requestSubmit();\n }\n\n clear() {\n this.inputs.forEach((input) => input.remove());\n }\n\n get inputs() {\n return this.element.querySelectorAll(\"input[data-generated]\");\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class SelectionFormController extends Controller {\n static values = {\n count: Number,\n primaryKey: { type: String, default: \"id\" },\n };\n static targets = [\"count\", \"singular\", \"plural\"];\n\n connect() {\n this.countValue = this.inputs.length;\n }\n\n /**\n * @param id to toggle\n * @return {boolean} true if selected, false if unselected\n */\n toggle(id) {\n const input = this.input(id);\n\n if (input) {\n input.remove();\n } else {\n this.element.insertAdjacentHTML(\n \"beforeend\",\n `<input type=\"hidden\" name=\"${this.primaryKeyValue}[]\" value=\"${id}\">`,\n );\n }\n\n this.countValue = this.visibleInputs.length;\n\n return !input;\n }\n\n /**\n * @param id to toggle visibility\n * @return {boolean} true if visible, false if not visible\n */\n visible(id, visible) {\n const input = this.input(id);\n\n if (input) {\n input.disabled = !visible;\n }\n\n this.countValue = this.visibleInputs.length;\n\n return !input;\n }\n\n /**\n * @returns {boolean} true if the given id is currently selected\n */\n isSelected(id) {\n return !!this.input(id);\n }\n\n get inputs() {\n return this.element.querySelectorAll(\n `input[name=\"${this.primaryKeyValue}[]\"]`,\n );\n }\n\n get visibleInputs() {\n return Array.from(this.inputs).filter((i) => !i.disabled);\n }\n\n input(id) {\n return this.element.querySelector(\n `input[name=\"${this.primaryKeyValue}[]\"][value=\"${id}\"]`,\n );\n }\n\n countValueChanged(count) {\n this.element.toggleAttribute(\"hidden\", count === 0);\n this.countTarget.textContent = count;\n this.singularTarget.toggleAttribute(\"hidden\", count !== 1);\n this.pluralTarget.toggleAttribute(\"hidden\", count === 1);\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class SelectionTableController extends Controller {\n static targets = [\"header\", \"item\"];\n static outlets = [\"tables--selection--form\"];\n\n itemTargetConnected(item) {\n this.update();\n }\n\n itemTargetDisconnected(item) {\n this.update();\n }\n\n toggleHeader(e) {\n this.items.forEach((item) => {\n if (item.checkedValue === e.target.checked) return;\n\n item.checkedValue = this.tablesSelectionFormOutlet.toggle(item.id);\n });\n }\n\n async update() {\n this.updating ||= Promise.resolve().then(() => {\n this.#update();\n delete this.updating;\n });\n\n return this.updating;\n }\n\n #update() {\n let present = 0;\n let checked = 0;\n\n this.items.forEach((item) => {\n present++;\n if (item.checkedValue) checked++;\n });\n\n this.headerInput.checked = present > 0 && checked === present;\n this.headerInput.indeterminate = checked > 0 && checked !== present;\n }\n\n get headerInput() {\n return this.headerTarget.querySelector(\"input\");\n }\n\n get items() {\n return this.itemTargets.map((el) => this.#itemOutlet(el)).filter((c) => c);\n }\n\n /**\n * Ideally we would be using outlets, but as of turbo 8.0.4 outlets do not fire disconnect events when morphing.\n *\n * Instead, we're using the targets to finds the controller.\n */\n #itemOutlet(el) {\n return this.application.getControllerForElementAndIdentifier(\n el,\n \"tables--selection--item\",\n );\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class QueryController extends Controller {\n static targets = [\"modal\"];\n\n disconnect() {\n delete this.pending;\n\n document.removeEventListener(\"selectionchange\", this.selection);\n }\n\n focus() {\n if (document.activeElement === this.query) return;\n\n this.query.addEventListener(\n \"focusin\",\n (e) => {\n e.target.setSelectionRange(-1, -1);\n },\n { once: true },\n );\n\n this.query.focus();\n }\n\n closeModal() {\n delete this.modalTarget.dataset.open;\n\n if (document.activeElement === this.query) document.activeElement.blur();\n\n document.removeEventListener(\"selectionchange\", this.selection);\n }\n\n openModal() {\n this.modalTarget.dataset.open = true;\n\n document.addEventListener(\"selectionchange\", this.selection);\n }\n\n /**\n * If the user presses escape once, clear the input.\n * If the user presses escape again, get them out of here.\n */\n clear() {\n if (this.query.value === \"\") {\n this.closeModal();\n } else {\n this.query.value = \"\";\n this.query.dispatchEvent(new Event(\"input\"));\n this.query.dispatchEvent(new Event(\"change\"));\n this.update();\n }\n }\n\n submit() {\n const hasFocus = this.isFocused;\n const position = hasFocus && this.query.selectionStart;\n\n if (this.pending) {\n clearTimeout(this.pending);\n delete this.pending;\n }\n\n // prevent an unnecessary `?q=` parameter from appearing in the URL\n if (this.query.value === \"\") {\n this.query.disabled = true;\n\n // restore input and focus after form submission\n setTimeout(() => {\n this.query.disabled = false;\n if (hasFocus) this.query.focus();\n }, 0);\n }\n\n // add/remove current cursor position\n if (hasFocus && position) {\n this.position.value = position;\n this.position.disabled = false;\n } else {\n this.position.value = \"\";\n this.position.disabled = true;\n }\n }\n\n update = () => {\n if (this.pending) clearTimeout(this.pending);\n this.pending = setTimeout(() => {\n this.element.requestSubmit();\n }, 300);\n };\n\n selection = () => {\n if (this.isFocused) this.update();\n };\n\n beforeMorphAttribute(e) {\n switch (e.detail.attributeName) {\n case \"data-open\":\n e.preventDefault();\n break;\n }\n }\n\n get query() {\n return this.element.querySelector(\"[role=searchbox]\");\n }\n\n get position() {\n return this.element.querySelector(\"input[name=p]\");\n }\n\n get isFocused() {\n return this.query === document.activeElement;\n }\n}\n"],"names":["DragState","constructor","list","event","id","this","cursorOffset","offsetY","initialPosition","target","offsetTop","targetId","updateCursor","row","callback","listOffset","getBoundingClientRect","top","itemPosition","clientY","updateItemPosition","updateScroll","previousScrollOffset","scrollDelta","position","Math","max","min","offsetHeight","SelectionItemController","Controller","static","params","Object","checked","Boolean","tablesSelectionFormOutletConnected","form","visible","checkedValue","isSelected","disconnect","hasTablesSelectionFormOutlet","tablesSelectionFormOutlet","change","e","preventDefault","toggle","paramsValue","paramsValueChanged","previous","update","checkedValueChanged","updating","Promise","resolve","then","element","querySelector","dispatch","detail","selected","Parser","tokens","values","parse","input","query","StringScanner","isEos","push","skipWhitespace","value","takeTagged","takeUntagged","token","scan","Token","matched","Untagged","key","valueAt","separator","takeArrayValue","takeSingleValue","Tagged","start","takeDelimiter","end","Array","takeQuotedValue","takeUnquotedValue","Value","render","document","createTextNode","span","createElement","className","innerText","super","appendChild","array","forEach","last","length","regex","match","exec","substring","index","Definitions","identifier","controllerConstructor","connect","from","parentElement","children","indexOf","id_value","dragUpdate","offset","dragOffset","style","zIndex","toggleAttribute","updateVisually","dragIndex","updateIndex","scope","id_name","index_name","name","reset","removeAttribute","hasChanges","index_value","round","comparisonIndex","startDragging","dragState","addEventListener","mousemove","mouseup","window","scroll","stopDragging","removeEventListener","tablesOrderableItemOutlets","item","drop","dragItem","newIndex","targetItem","insertAdjacentElement","commitChanges","tablesOrderableFormOutlet","clear","add","submit","mousedown","isDragging","animate","ticking","requestAnimationFrame","tablesOrderableFormOutlets","tablesOrderableFormOutletConnected","tablesOrderableFormOutletDisconnected","currentItems","find","toSorted","a","b","String","scopeValue","insertAdjacentHTML","inputs","requestSubmit","remove","querySelectorAll","count","Number","primaryKey","type","default","countValue","primaryKeyValue","visibleInputs","disabled","filter","i","countValueChanged","countTarget","textContent","singularTarget","pluralTarget","itemTargetConnected","itemTargetDisconnected","toggleHeader","items","present","headerInput","indeterminate","headerTarget","itemTargets","map","el","itemOutlet","c","application","getControllerForElementAndIdentifier","pending","selection","focus","activeElement","setSelectionRange","once","closeModal","modalTarget","dataset","open","blur","openModal","dispatchEvent","Event","hasFocus","isFocused","selectionStart","clearTimeout","setTimeout","beforeMorphAttribute","attributeName","queryValue","inputTarget","connected","queryValueChanged","highlightTarget","innerHTML"],"mappings":"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"}
|
1
|
+
{"version":3,"file":"tables.min.js","sources":["../../../javascript/tables/orderable/list_controller.js","../../../javascript/tables/selection/item_controller.js","../../../javascript/tables/query_input_controller.js","../../../javascript/tables/application.js","../../../javascript/tables/orderable/item_controller.js","../../../javascript/tables/orderable/form_controller.js","../../../javascript/tables/selection/form_controller.js","../../../javascript/tables/selection/table_controller.js","../../../javascript/tables/query_controller.js"],"sourcesContent":["import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableListController extends Controller {\n static outlets = [\"tables--orderable--item\", \"tables--orderable--form\"];\n\n //region State transitions\n\n startDragging(dragState) {\n this.dragState = dragState;\n\n document.addEventListener(\"mousemove\", this.mousemove);\n document.addEventListener(\"mouseup\", this.mouseup);\n window.addEventListener(\"scroll\", this.scroll, true);\n\n this.element.style.position = \"relative\";\n }\n\n stopDragging() {\n const dragState = this.dragState;\n delete this.dragState;\n\n document.removeEventListener(\"mousemove\", this.mousemove);\n document.removeEventListener(\"mouseup\", this.mouseup);\n window.removeEventListener(\"scroll\", this.scroll, true);\n\n this.element.removeAttribute(\"style\");\n this.tablesOrderableItemOutlets.forEach((item) => item.reset());\n\n return dragState;\n }\n\n drop() {\n // note: early returns guard against turbo updates that prevent us finding\n // the right item to drop on. In this situation it's better to discard the\n // drop than to drop in the wrong place.\n\n const dragItem = this.dragItem;\n\n if (!dragItem) return;\n\n const newIndex = dragItem.dragIndex;\n const targetItem = this.tablesOrderableItemOutlets[newIndex];\n\n if (!targetItem) return;\n\n // swap the dragged item into the correct position for its current offset\n if (newIndex < dragItem.index) {\n targetItem.row.insertAdjacentElement(\"beforebegin\", dragItem.row);\n } else if (newIndex > dragItem.index) {\n targetItem.row.insertAdjacentElement(\"afterend\", dragItem.row);\n }\n\n // reindex all items based on their new positions\n this.tablesOrderableItemOutlets.forEach((item, index) =>\n item.updateIndex(index),\n );\n\n // save the changes\n this.commitChanges();\n }\n\n commitChanges() {\n // clear any existing inputs to prevent duplicates\n this.tablesOrderableFormOutlet.clear();\n\n // insert any items that have changed position\n this.tablesOrderableItemOutlets.forEach((item) => {\n if (item.hasChanges) this.tablesOrderableFormOutlet.add(item);\n });\n\n this.tablesOrderableFormOutlet.submit();\n }\n\n //endregion\n\n //region Events\n\n mousedown(event) {\n if (this.isDragging) return;\n\n const target = this.#targetItem(event.target);\n\n if (!target) return;\n\n event.preventDefault(); // prevent built-in drag\n\n this.startDragging(new DragState(this.element, event, target.id));\n\n this.dragState.updateCursor(this.element, target.row, event, this.animate);\n }\n\n mousemove = (event) => {\n if (!this.isDragging) return;\n\n event.preventDefault(); // prevent build-in drag\n\n if (this.ticking) return;\n\n this.ticking = true;\n\n window.requestAnimationFrame(() => {\n this.ticking = false;\n this.dragState.updateCursor(\n this.element,\n this.dragItem.row,\n event,\n this.animate,\n );\n });\n };\n\n scroll = (event) => {\n if (!this.isDragging || this.ticking) return;\n\n this.ticking = true;\n\n window.requestAnimationFrame(() => {\n this.ticking = false;\n this.dragState.updateScroll(\n this.element,\n this.dragItem.row,\n this.animate,\n );\n });\n };\n\n mouseup = (event) => {\n if (!this.isDragging) return;\n\n this.drop();\n this.stopDragging();\n this.tablesOrderableFormOutlets.forEach((form) => delete form.dragState);\n };\n\n tablesOrderableFormOutletConnected(form, element) {\n if (form.dragState) {\n // restore the previous controller's state\n this.startDragging(form.dragState);\n }\n }\n\n tablesOrderableFormOutletDisconnected(form, element) {\n if (this.isDragging) {\n // cache drag state in the form\n form.dragState = this.stopDragging();\n }\n }\n\n //endregion\n\n //region Helpers\n\n /**\n * Updates the position of the drag item with a relative offset. Updates\n * other items relative to the new position of the drag item, as required.\n *\n * @callback {OrderableListController~animate}\n * @param {number} offset\n */\n animate = (offset) => {\n const dragItem = this.dragItem;\n\n // Visually update the dragItem so it follows the cursor\n dragItem.dragUpdate(offset);\n\n // Visually updates the position of all items in the list relative to the\n // dragged item. No actual changes to orderings at this stage.\n this.#currentItems.forEach((item, index) => {\n if (item === dragItem) return;\n item.updateVisually(index);\n });\n };\n\n get isDragging() {\n return !!this.dragState;\n }\n\n get dragItem() {\n if (!this.isDragging) return null;\n\n return this.tablesOrderableItemOutlets.find(\n (item) => item.id === this.dragState.targetId,\n );\n }\n\n /**\n * Returns the current items in the list, sorted by their current index.\n * Current uses the drag index if the item is being dragged, if set.\n *\n * @returns {Array[OrderableRowController]}\n */\n get #currentItems() {\n return this.tablesOrderableItemOutlets.toSorted(\n (a, b) => a.comparisonIndex - b.comparisonIndex,\n );\n }\n\n /**\n * Returns the item outlet that was clicked on, if any.\n *\n * @param element {HTMLElement} the clicked ordinal cell\n * @returns {OrderableRowController}\n */\n #targetItem(element) {\n return this.tablesOrderableItemOutlets.find(\n (item) => item.element === element,\n );\n }\n\n //endregion\n}\n\n/**\n * During drag we want to be able to translate a document-relative coordinate\n * into a coordinate relative to the list element. This state object calculates\n * and stores internal state so that we can translate absolute page coordinates\n * from mouse events into relative offsets for the list items within the list\n * element.\n *\n * We also keep track of the drag target so that if the controller is attached\n * to a new element during the drag we can continue after the turbo update.\n */\nclass DragState {\n /**\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param event {MouseEvent} the initial event\n * @param id {String} the id of the element being dragged\n */\n constructor(list, event, id) {\n // cursor offset is the offset of the cursor relative to the drag item\n this.cursorOffset = event.offsetY;\n\n // initial offset is the offset position of the drag item at drag start\n this.initialPosition = event.target.offsetTop - list.offsetTop;\n\n // id of the item being dragged\n this.targetId = id;\n }\n\n /**\n * Calculates the offset of the drag item relative to its initial position.\n *\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param row {HTMLElement} the row being dragged\n * @param event {MouseEvent} the current event\n * @param callback {OrderableListController~animate} updates the drag item with a relative offset\n */\n updateCursor(list, row, event, callback) {\n // Calculate and store the list offset relative to the viewport\n // This value is cached so we can calculate the outcome of any scroll events\n this.listOffset = list.getBoundingClientRect().top;\n\n // Calculate the position of the cursor relative to the list.\n // Accounts for scroll offsets by using the item's bounding client rect.\n const cursorPosition = event.clientY - this.listOffset;\n\n // intended item position relative to the list, from cursor position\n let itemPosition = cursorPosition - this.cursorOffset;\n\n this.#updateItemPosition(list, row, itemPosition, callback);\n }\n\n /**\n * Animates the item's position as the list scrolls. Requires a previous call\n * to set the scroll offset.\n *\n * @param list {HTMLElement} the list controller's element (tbody)\n * @param row {HTMLElement} the row being dragged\n * @param callback {OrderableListController~animate} updates the drag item with a relative offset\n */\n updateScroll(list, row, callback) {\n const previousScrollOffset = this.listOffset;\n\n // Calculate and store the list offset relative to the viewport\n // This value is cached so we can calculate the outcome of any scroll events\n this.listOffset = list.getBoundingClientRect().top;\n\n // Calculate the change in scroll offset since the last update\n const scrollDelta = previousScrollOffset - this.listOffset;\n\n // intended item position relative to the list, from cursor position\n const position = this.position + scrollDelta;\n\n this.#updateItemPosition(list, row, position, callback);\n }\n\n #updateItemPosition(list, row, position, callback) {\n // ensure itemPosition is within the bounds of the list (tbody)\n position = Math.max(position, 0);\n position = Math.min(position, list.offsetHeight - row.offsetHeight);\n\n // cache the item's position relative to the list for use in scroll events\n this.position = position;\n\n // Item has position: relative, so we want to calculate the amount to move\n // the item relative to it's DOM position to represent how much it has been\n // dragged by.\n const offset = position - this.initialPosition;\n\n // Convert itemPosition from offset relative to list to offset relative to\n // its position within the DOM (if it hadn't moved).\n callback(offset);\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\n/**\n * Couples an input element in a row to the selection form which is turbo-permanent and outside the table.\n * When the input is toggled, the form will create/destroy hidden inputs. The checkbox inside this cell will follow\n * the hidden inputs.\n *\n * Cell value may change when:\n * * cell connects, e.g. when the user paginates\n * * cell is re-used by turbo-morph, e.g. pagination\n * * cell is toggled\n * * select-all/de-select-all on table header\n */\nexport default class SelectionItemController extends Controller {\n static outlets = [\"tables--selection--form\"];\n static values = {\n params: Object,\n checked: Boolean,\n };\n\n tablesSelectionFormOutletConnected(form) {\n form.visible(this.id, true);\n this.checkedValue = form.isSelected(this.id);\n }\n\n disconnect() {\n // Remove from form's list of visible selections.\n // This should be an outlet disconnect, but those events are not reliable in turbo 8.0\n if (this.hasTablesSelectionFormOutlet) {\n this.tablesSelectionFormOutlet.visible(this.id, false);\n }\n }\n\n change(e) {\n e.preventDefault();\n\n this.checkedValue = this.tablesSelectionFormOutlet.toggle(this.id);\n }\n\n get id() {\n return this.paramsValue.id;\n }\n\n /**\n * Update checked to match match selection form. This occurs when the item is re-used by turbo-morph.\n */\n paramsValueChanged(params, previous) {\n if (!this.hasTablesSelectionFormOutlet) return;\n\n // if id is changing (e.g. morph) then let the form know that the previous id is now not visible\n if (previous.id !== params.id) {\n this.tablesSelectionFormOutlet.visible(previous.id, false);\n }\n\n // tell form that our id is now visible in the table\n this.tablesSelectionFormOutlet.visible(params.id, true);\n\n // id has changed, so update checked from form\n this.checkedValue = this.tablesSelectionFormOutlet.isSelected(params.id);\n\n // propagate changes\n this.update();\n }\n\n /**\n * Update input to match checked. This occurs when the item is toggled, connected, or morphed.\n */\n checkedValueChanged() {\n if (!this.hasTablesSelectionFormOutlet) return;\n\n // ensure that checked matches the form, i.e. if morphed\n this.checkedValue = this.tablesSelectionFormOutlet.isSelected(this.id);\n\n // propagate changes\n this.update();\n }\n\n /**\n * Notify table that id or value may have changed. Note that this may fire when nothing has changed.\n *\n * Debouncing to minimise dom updates.\n */\n async update() {\n this.updating ||= Promise.resolve().then(() => {\n this.#update();\n delete this.updating;\n });\n\n return this.updating;\n }\n\n #update() {\n this.element.querySelector(\"input\").checked = this.checkedValue;\n this.dispatch(\"select\", {\n detail: { id: this.id, selected: this.checkedValue },\n });\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class QueryInputController extends Controller {\n static targets = [\"input\", \"highlight\"];\n static values = { query: String };\n\n connect() {\n this.queryValue = this.inputTarget.value;\n this.element.dataset.connected = \"\";\n }\n\n disconnect() {\n delete this.element.dataset.connected;\n }\n\n update() {\n this.queryValue = this.inputTarget.value;\n }\n\n queryValueChanged(query) {\n this.highlightTarget.innerHTML = \"\";\n\n new Parser().parse(query).tokens.forEach((token) => {\n this.highlightTarget.appendChild(token.render());\n });\n }\n}\n\nclass Parser {\n constructor() {\n this.tokens = [];\n this.values = null;\n }\n\n parse(input) {\n const query = new StringScanner(input);\n\n while (!query.isEos()) {\n this.push(this.skipWhitespace(query));\n\n const value = this.takeTagged(query) || this.takeUntagged(query);\n\n if (!this.push(value)) break;\n }\n\n return this;\n }\n\n push(token) {\n if (token) {\n this.values ? this.values.push(token) : this.tokens.push(token);\n }\n\n return !!token;\n }\n\n skipWhitespace(query) {\n if (!query.scan(/\\s+/)) return;\n\n return new Token(query.matched());\n }\n\n takeUntagged(query) {\n if (!query.scan(/\\S+/)) return;\n\n return new Untagged(query.matched());\n }\n\n takeTagged(query) {\n if (!query.scan(/(\\w+(?:\\.\\w+)?)(:\\s*)/)) return;\n\n const key = query.valueAt(1);\n const separator = query.valueAt(2);\n\n const value =\n this.takeArrayValue(query) || this.takeSingleValue(query) || new Token();\n\n return new Tagged(key, separator, value);\n }\n\n takeArrayValue(query) {\n if (!query.scan(/\\[\\s*/)) return;\n\n const start = new Token(query.matched());\n const values = (this.values = []);\n\n while (!query.isEos()) {\n if (!this.push(this.takeSingleValue(query))) break;\n if (!this.push(this.takeDelimiter(query))) break;\n }\n\n query.scan(/\\s*]/);\n const end = new Token(query.matched());\n\n this.values = null;\n\n return new Array(start, values, end);\n }\n\n takeDelimiter(query) {\n if (!query.scan(/\\s*,\\s*/)) return;\n\n return new Token(query.matched());\n }\n\n takeSingleValue(query) {\n return this.takeQuotedValue(query) || this.takeUnquotedValue(query);\n }\n\n takeQuotedValue(query) {\n if (!query.scan(/\"([^\"]*)\"/)) return;\n\n return new Value(query.matched());\n }\n\n takeUnquotedValue(query) {\n if (!query.scan(/[^ \\],]*/)) return;\n\n return new Value(query.matched());\n }\n}\n\nclass Token {\n constructor(value = \"\") {\n this.value = value;\n }\n\n render() {\n return document.createTextNode(this.value);\n }\n}\n\nclass Value extends Token {\n render() {\n const span = document.createElement(\"span\");\n span.className = \"value\";\n span.innerText = this.value;\n\n return span;\n }\n}\n\nclass Tagged extends Token {\n constructor(key, separator, value) {\n super();\n\n this.key = key;\n this.separator = separator;\n this.value = value;\n }\n\n render() {\n const span = document.createElement(\"span\");\n span.className = \"tag\";\n\n const key = document.createElement(\"span\");\n key.className = \"key\";\n key.innerText = this.key;\n\n span.appendChild(key);\n span.appendChild(document.createTextNode(this.separator));\n span.appendChild(this.value.render());\n\n return span;\n }\n}\n\nclass Untagged extends Token {\n render() {\n const span = document.createElement(\"span\");\n span.className = \"untagged\";\n span.innerText = this.value;\n return span;\n }\n}\n\nclass Array extends Token {\n constructor(start, values, end) {\n super();\n\n this.start = start;\n this.values = values;\n this.end = end;\n }\n\n render() {\n const array = document.createElement(\"span\");\n array.className = \"array-values\";\n array.appendChild(this.start.render());\n\n this.values.forEach((value) => {\n const span = document.createElement(\"span\");\n span.appendChild(value.render());\n array.appendChild(span);\n });\n\n array.appendChild(this.end.render());\n\n return array;\n }\n}\n\nclass StringScanner {\n constructor(input) {\n this.input = input;\n this.position = 0;\n this.last = null;\n }\n\n isEos() {\n return this.position >= this.input.length;\n }\n\n scan(regex) {\n const match = regex.exec(this.input.substring(this.position));\n if (match?.index === 0) {\n this.last = match;\n this.position += match[0].length;\n return true;\n } else {\n this.last = {};\n return false;\n }\n }\n\n matched() {\n return this.last && this.last[0];\n }\n\n valueAt(index) {\n return this.last && this.last[index];\n }\n}\n","import OrderableItemController from \"./orderable/item_controller\";\nimport OrderableListController from \"./orderable/list_controller\";\nimport OrderableFormController from \"./orderable/form_controller\";\nimport SelectionFormController from \"./selection/form_controller\";\nimport SelectionItemController from \"./selection/item_controller\";\nimport SelectionTableController from \"./selection/table_controller\";\nimport QueryController from \"./query_controller\";\nimport QueryInputController from \"./query_input_controller\";\n\nconst Definitions = [\n {\n identifier: \"tables--orderable--item\",\n controllerConstructor: OrderableItemController,\n },\n {\n identifier: \"tables--orderable--list\",\n controllerConstructor: OrderableListController,\n },\n {\n identifier: \"tables--orderable--form\",\n controllerConstructor: OrderableFormController,\n },\n {\n identifier: \"tables--selection--form\",\n controllerConstructor: SelectionFormController,\n },\n {\n identifier: \"tables--selection--item\",\n controllerConstructor: SelectionItemController,\n },\n {\n identifier: \"tables--selection--table\",\n controllerConstructor: SelectionTableController,\n },\n {\n identifier: \"tables--query\",\n controllerConstructor: QueryController,\n },\n {\n identifier: \"tables--query-input\",\n controllerConstructor: QueryInputController,\n },\n];\n\nexport { Definitions as default };\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableRowController extends Controller {\n static values = {\n params: Object,\n };\n\n connect() {\n // index from server may be inconsistent with the visual ordering,\n // especially if this is a new node. Use positional indexes instead,\n // as these are the values we will send on save.\n this.index = domIndex(this.row);\n }\n\n paramsValueChanged(params) {\n this.id = params.id_value;\n }\n\n dragUpdate(offset) {\n this.dragOffset = offset;\n this.row.style.position = \"relative\";\n this.row.style.top = offset + \"px\";\n this.row.style.zIndex = \"1\";\n this.row.toggleAttribute(\"dragging\", true);\n }\n\n /**\n * Called on items that are not the dragged item during drag. Updates the\n * visual position of the item relative to the dragged item.\n *\n * @param index {number} intended index of the item during drag\n */\n updateVisually(index) {\n this.row.style.position = \"relative\";\n this.row.style.top = `${\n this.row.offsetHeight * (index - this.dragIndex)\n }px`;\n }\n\n /**\n * Set the index value of the item. This is called on all items after a drop\n * event. If the index is different to the params index then this item has\n * changed.\n *\n * @param index {number} the new index value\n */\n updateIndex(index) {\n this.index = index;\n }\n\n /** Retrieve params for use in the form */\n params(scope) {\n const { id_name, id_value, index_name } = this.paramsValue;\n return [\n { name: `${scope}[${id_value}][${id_name}]`, value: this.id },\n { name: `${scope}[${id_value}][${index_name}]`, value: this.index },\n ];\n }\n\n /**\n * Restore any visual changes made during drag and remove the drag state.\n */\n reset() {\n delete this.dragOffset;\n this.row.removeAttribute(\"style\");\n this.row.removeAttribute(\"dragging\");\n }\n\n /**\n * @returns {boolean} true when the item has a change to its index value\n */\n get hasChanges() {\n return this.paramsValue.index_value !== this.index;\n }\n\n /**\n * Calculate the relative index of the item during drag. This is used to\n * sort items during drag as it takes into account any uncommitted changes\n * to index caused by the drag offset.\n *\n * @returns {number} index for the purposes of drag and drop ordering\n */\n get dragIndex() {\n if (this.dragOffset && this.dragOffset !== 0) {\n return this.index + Math.round(this.dragOffset / this.row.offsetHeight);\n } else {\n return this.index;\n }\n }\n\n /**\n * Index value for use in comparisons during drag. This is used to determine\n * whether the dragged item is above or below another item. If this item is\n * being dragged then we offset the index by 0.5 to ensure that it jumps up\n * or down when it reaches the midpoint of the item above or below it.\n *\n * @returns {number}\n */\n get comparisonIndex() {\n if (this.dragOffset) {\n return this.dragIndex + (this.dragOffset > 0 ? 0.5 : -0.5);\n } else {\n return this.index;\n }\n }\n\n /**\n * The containing row element.\n *\n * @returns {HTMLElement}\n */\n get row() {\n return this.element.parentElement;\n }\n}\n\nfunction domIndex(element) {\n return Array.from(element.parentElement.children).indexOf(element);\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class OrderableFormController extends Controller {\n static values = { scope: String };\n\n add(item) {\n item.params(this.scopeValue).forEach(({ name, value }) => {\n this.element.insertAdjacentHTML(\n \"beforeend\",\n `<input type=\"hidden\" name=\"${name}\" value=\"${value}\" data-generated>`,\n );\n });\n }\n\n submit() {\n if (this.inputs.length === 0) return;\n\n this.element.requestSubmit();\n }\n\n clear() {\n this.inputs.forEach((input) => input.remove());\n }\n\n get inputs() {\n return this.element.querySelectorAll(\"input[data-generated]\");\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class SelectionFormController extends Controller {\n static values = {\n count: Number,\n primaryKey: { type: String, default: \"id\" },\n };\n static targets = [\"count\", \"singular\", \"plural\"];\n\n connect() {\n this.countValue = this.inputs.length;\n }\n\n /**\n * @param id to toggle\n * @return {boolean} true if selected, false if unselected\n */\n toggle(id) {\n const input = this.input(id);\n\n if (input) {\n input.remove();\n } else {\n this.element.insertAdjacentHTML(\n \"beforeend\",\n `<input type=\"hidden\" name=\"${this.primaryKeyValue}[]\" value=\"${id}\">`,\n );\n }\n\n this.countValue = this.visibleInputs.length;\n\n return !input;\n }\n\n /**\n * @param id to toggle visibility\n * @return {boolean} true if visible, false if not visible\n */\n visible(id, visible) {\n const input = this.input(id);\n\n if (input) {\n input.disabled = !visible;\n }\n\n this.countValue = this.visibleInputs.length;\n\n return !input;\n }\n\n /**\n * @returns {boolean} true if the given id is currently selected\n */\n isSelected(id) {\n return !!this.input(id);\n }\n\n get inputs() {\n return this.element.querySelectorAll(\n `input[name=\"${this.primaryKeyValue}[]\"]`,\n );\n }\n\n get visibleInputs() {\n return Array.from(this.inputs).filter((i) => !i.disabled);\n }\n\n input(id) {\n return this.element.querySelector(\n `input[name=\"${this.primaryKeyValue}[]\"][value=\"${id}\"]`,\n );\n }\n\n countValueChanged(count) {\n this.element.toggleAttribute(\"hidden\", count === 0);\n this.countTarget.textContent = count;\n this.singularTarget.toggleAttribute(\"hidden\", count !== 1);\n this.pluralTarget.toggleAttribute(\"hidden\", count === 1);\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class SelectionTableController extends Controller {\n static targets = [\"header\", \"item\"];\n static outlets = [\"tables--selection--form\"];\n\n itemTargetConnected(item) {\n this.update();\n }\n\n itemTargetDisconnected(item) {\n this.update();\n }\n\n toggleHeader(e) {\n this.items.forEach((item) => {\n if (item.checkedValue === e.target.checked) return;\n\n item.checkedValue = this.tablesSelectionFormOutlet.toggle(item.id);\n });\n }\n\n async update() {\n this.updating ||= Promise.resolve().then(() => {\n this.#update();\n delete this.updating;\n });\n\n return this.updating;\n }\n\n #update() {\n let present = 0;\n let checked = 0;\n\n this.items.forEach((item) => {\n present++;\n if (item.checkedValue) checked++;\n });\n\n this.headerInput.checked = present > 0 && checked === present;\n this.headerInput.indeterminate = checked > 0 && checked !== present;\n }\n\n get headerInput() {\n return this.headerTarget.querySelector(\"input\");\n }\n\n get items() {\n return this.itemTargets.map((el) => this.#itemOutlet(el)).filter((c) => c);\n }\n\n /**\n * Ideally we would be using outlets, but as of turbo 8.0.4 outlets do not fire disconnect events when morphing.\n *\n * Instead, we're using the targets to finds the controller.\n */\n #itemOutlet(el) {\n return this.application.getControllerForElementAndIdentifier(\n el,\n \"tables--selection--item\",\n );\n }\n}\n","import { Controller } from \"@hotwired/stimulus\";\n\nexport default class QueryController extends Controller {\n static targets = [\"modal\"];\n\n disconnect() {\n delete this.pending;\n\n document.removeEventListener(\"selectionchange\", this.selection);\n }\n\n focus() {\n if (document.activeElement === this.query) return;\n\n this.query.addEventListener(\n \"focusin\",\n (e) => {\n e.target.setSelectionRange(-1, -1);\n },\n { once: true },\n );\n\n this.query.focus();\n }\n\n closeModal() {\n delete this.modalTarget.dataset.open;\n\n if (document.activeElement === this.query) document.activeElement.blur();\n\n document.removeEventListener(\"selectionchange\", this.selection);\n }\n\n openModal() {\n this.modalTarget.dataset.open = true;\n\n document.addEventListener(\"selectionchange\", this.selection);\n }\n\n /**\n * If the user presses escape once, clear the input.\n * If the user presses escape again, get them out of here.\n */\n clear() {\n if (this.query.value === \"\") {\n this.closeModal();\n } else {\n this.query.value = \"\";\n this.query.dispatchEvent(new Event(\"input\"));\n this.query.dispatchEvent(new Event(\"change\"));\n this.update();\n }\n }\n\n submit() {\n const hasFocus = this.isFocused;\n const position = hasFocus && this.query.selectionStart;\n\n if (this.pending) {\n clearTimeout(this.pending);\n delete this.pending;\n }\n\n // prevent an unnecessary `?q=` parameter from appearing in the URL\n if (this.query.value === \"\") {\n this.query.disabled = true;\n\n // restore input and focus after form submission\n setTimeout(() => {\n this.query.disabled = false;\n if (hasFocus) this.query.focus();\n }, 0);\n }\n\n // add/remove current cursor position\n if (hasFocus && position) {\n this.position.value = position;\n this.position.disabled = false;\n } else {\n this.position.value = \"\";\n this.position.disabled = true;\n }\n }\n\n update = () => {\n if (this.pending) clearTimeout(this.pending);\n this.pending = setTimeout(() => {\n this.element.requestSubmit();\n }, 300);\n };\n\n selection = () => {\n if (this.isFocused && this.query.value.length > 0) this.update();\n };\n\n beforeMorphAttribute(e) {\n switch (e.detail.attributeName) {\n case \"data-open\":\n e.preventDefault();\n break;\n }\n }\n\n get query() {\n return this.element.querySelector(\"[role=searchbox]\");\n }\n\n get position() {\n return this.element.querySelector(\"input[name=p]\");\n }\n\n get isFocused() {\n return this.query === document.activeElement;\n }\n}\n"],"names":["DragState","constructor","list","event","id","this","cursorOffset","offsetY","initialPosition","target","offsetTop","targetId","updateCursor","row","callback","listOffset","getBoundingClientRect","top","itemPosition","clientY","updateItemPosition","updateScroll","previousScrollOffset","scrollDelta","position","Math","max","min","offsetHeight","SelectionItemController","Controller","static","params","Object","checked","Boolean","tablesSelectionFormOutletConnected","form","visible","checkedValue","isSelected","disconnect","hasTablesSelectionFormOutlet","tablesSelectionFormOutlet","change","e","preventDefault","toggle","paramsValue","paramsValueChanged","previous","update","checkedValueChanged","updating","Promise","resolve","then","element","querySelector","dispatch","detail","selected","Parser","tokens","values","parse","input","query","StringScanner","isEos","push","skipWhitespace","value","takeTagged","takeUntagged","token","scan","Token","matched","Untagged","key","valueAt","separator","takeArrayValue","takeSingleValue","Tagged","start","takeDelimiter","end","Array","takeQuotedValue","takeUnquotedValue","Value","render","document","createTextNode","span","createElement","className","innerText","super","appendChild","array","forEach","last","length","regex","match","exec","substring","index","Definitions","identifier","controllerConstructor","connect","from","parentElement","children","indexOf","id_value","dragUpdate","offset","dragOffset","style","zIndex","toggleAttribute","updateVisually","dragIndex","updateIndex","scope","id_name","index_name","name","reset","removeAttribute","hasChanges","index_value","round","comparisonIndex","startDragging","dragState","addEventListener","mousemove","mouseup","window","scroll","stopDragging","removeEventListener","tablesOrderableItemOutlets","item","drop","dragItem","newIndex","targetItem","insertAdjacentElement","commitChanges","tablesOrderableFormOutlet","clear","add","submit","mousedown","isDragging","animate","ticking","requestAnimationFrame","tablesOrderableFormOutlets","tablesOrderableFormOutletConnected","tablesOrderableFormOutletDisconnected","currentItems","find","toSorted","a","b","String","scopeValue","insertAdjacentHTML","inputs","requestSubmit","remove","querySelectorAll","count","Number","primaryKey","type","default","countValue","primaryKeyValue","visibleInputs","disabled","filter","i","countValueChanged","countTarget","textContent","singularTarget","pluralTarget","itemTargetConnected","itemTargetDisconnected","toggleHeader","items","present","headerInput","indeterminate","headerTarget","itemTargets","map","el","itemOutlet","c","application","getControllerForElementAndIdentifier","pending","selection","focus","activeElement","setSelectionRange","once","closeModal","modalTarget","dataset","open","blur","openModal","dispatchEvent","Event","hasFocus","isFocused","selectionStart","clearTimeout","setTimeout","beforeMorphAttribute","attributeName","queryValue","inputTarget","connected","queryValueChanged","highlightTarget","innerHTML"],"mappings":"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"}
|
@@ -96,10 +96,12 @@
|
|
96
96
|
}
|
97
97
|
}
|
98
98
|
|
99
|
-
|
99
|
+
footer {
|
100
100
|
grid-area: footer;
|
101
101
|
display: flex;
|
102
|
-
justify-content: flex-
|
103
|
-
padding-
|
102
|
+
justify-content: flex-start;
|
103
|
+
padding-inline: 1rem;
|
104
|
+
padding-block: 0.5rem;
|
105
|
+
border-top: 1px solid #d3d3d3;
|
104
106
|
}
|
105
107
|
}
|
@@ -7,7 +7,11 @@ module Katalyst
|
|
7
7
|
# Expect to see NoMethodError failures if pagy is not available
|
8
8
|
"Pagy::Frontend".safe_constantize&.tap { |pagy| include(pagy) }
|
9
9
|
|
10
|
-
|
10
|
+
def self.pagy_legacy?
|
11
|
+
Pagy::VERSION.scan(/\d+/).first.to_i <= 8
|
12
|
+
end
|
13
|
+
|
14
|
+
delegate :pagy_legacy?, to: :class
|
11
15
|
|
12
16
|
def initialize(collection: nil, pagy: nil, **pagy_options)
|
13
17
|
super()
|
@@ -24,9 +28,19 @@ module Katalyst
|
|
24
28
|
pagy_nav(@pagy, **pagy_options).html_safe # rubocop:disable Rails/OutputSafety
|
25
29
|
end
|
26
30
|
|
31
|
+
def pagy_options
|
32
|
+
default_pagy_options.merge(@pagy_options)
|
33
|
+
end
|
34
|
+
|
27
35
|
def inspect
|
28
36
|
"#<#{self.class.name} pagy: #{@pagy.inspect}>"
|
29
37
|
end
|
38
|
+
|
39
|
+
private
|
40
|
+
|
41
|
+
def default_pagy_options
|
42
|
+
pagy_legacy? ? {} : { anchor_string: 'data-turbo-action="replace"' }
|
43
|
+
end
|
30
44
|
end
|
31
45
|
end
|
32
46
|
end
|
@@ -22,6 +22,12 @@ module Katalyst
|
|
22
22
|
collection.suggestions.each do |suggestion|
|
23
23
|
with_suggestion(suggestion:)
|
24
24
|
end
|
25
|
+
|
26
|
+
unless footer?
|
27
|
+
with_footer do
|
28
|
+
link_to("Search syntax", "https://katalyst.github.io/tables/users/queries")
|
29
|
+
end
|
30
|
+
end
|
25
31
|
end
|
26
32
|
|
27
33
|
private
|
@@ -1,10 +1,4 @@
|
|
1
|
-
<%= form_with(method: :patch,
|
2
|
-
id:,
|
3
|
-
class: "tables--selection--form",
|
4
|
-
data: { controller: form_controller,
|
5
|
-
turbo_action: "replace",
|
6
|
-
turbo_permanent: "" },
|
7
|
-
html: { action: false, hidden: "" }) do |form| %>
|
1
|
+
<%= form_with(method: :patch, **html_attributes) do |form| %>
|
8
2
|
<p class="tables--selection--summary">
|
9
3
|
<span data-<%= form_target("count") %>>0</span>
|
10
4
|
<span data-<%= form_target("singular") %> hidden><%= singular_name %></span>
|
@@ -4,6 +4,7 @@ module Katalyst
|
|
4
4
|
module Tables
|
5
5
|
module Selectable
|
6
6
|
class FormComponent < ViewComponent::Base # :nodoc:
|
7
|
+
include Katalyst::HtmlAttributes
|
7
8
|
include Katalyst::Tables::Identifiable::Defaults
|
8
9
|
|
9
10
|
attr_reader :id, :primary_key
|
@@ -14,7 +15,7 @@ module Katalyst
|
|
14
15
|
def initialize(collection:,
|
15
16
|
id: nil,
|
16
17
|
primary_key: :id)
|
17
|
-
super
|
18
|
+
super()
|
18
19
|
|
19
20
|
@collection = collection
|
20
21
|
@id = id || Selectable.default_form_id(collection)
|
@@ -27,6 +28,18 @@ module Katalyst
|
|
27
28
|
|
28
29
|
private
|
29
30
|
|
31
|
+
def default_html_attributes
|
32
|
+
{
|
33
|
+
id:,
|
34
|
+
class: "tables--selection--form",
|
35
|
+
data: {
|
36
|
+
controller: form_controller,
|
37
|
+
turbo_action: "replace",
|
38
|
+
},
|
39
|
+
html: { action: false, hidden: "" },
|
40
|
+
}
|
41
|
+
end
|
42
|
+
|
30
43
|
def form_controller
|
31
44
|
FORM_CONTROLLER
|
32
45
|
end
|
@@ -8,6 +8,7 @@ module Katalyst
|
|
8
8
|
|
9
9
|
included do
|
10
10
|
class_attribute :_default_table_component, instance_accessor: false
|
11
|
+
class_attribute :_default_table_pagination_component, instance_accessor: false
|
11
12
|
class_attribute :_default_table_query_component, instance_accessor: false
|
12
13
|
class_attribute :_default_summary_table_component, instance_accessor: false
|
13
14
|
end
|
@@ -21,6 +22,14 @@ module Katalyst
|
|
21
22
|
self._default_table_component = component
|
22
23
|
end
|
23
24
|
|
25
|
+
# Set the table pagination component to be used as the default for all tables
|
26
|
+
# in the views rendered by this controller and its subclasses.
|
27
|
+
#
|
28
|
+
# @param component [Class] the table pagination component class to use
|
29
|
+
def default_table_pagination_component(component)
|
30
|
+
self._default_table_pagination_component = component
|
31
|
+
end
|
32
|
+
|
24
33
|
# Set the table query component to be used as the default for all tables
|
25
34
|
# in the views rendered by this controller and its subclasses.
|
26
35
|
#
|
@@ -44,6 +53,11 @@ module Katalyst
|
|
44
53
|
self.class._default_table_component
|
45
54
|
end
|
46
55
|
|
56
|
+
# Default table pagination component for this controller
|
57
|
+
def default_table_pagination_component
|
58
|
+
self.class._default_table_pagination_component
|
59
|
+
end
|
60
|
+
|
47
61
|
# Default table query component for this controller
|
48
62
|
def default_table_query_component
|
49
63
|
self.class._default_table_query_component
|
@@ -55,6 +55,14 @@ module Katalyst
|
|
55
55
|
render(Selectable::FormComponent.new(collection:, id:, primary_key:), &)
|
56
56
|
end
|
57
57
|
|
58
|
+
# Construct pagination navigation for the current page. Defaults to pagy_nav.
|
59
|
+
#
|
60
|
+
# @param collection [Katalyst::Tables::Collection::Core] the collection to render
|
61
|
+
def table_pagination_with(collection:, **)
|
62
|
+
component ||= default_table_pagination_component_class
|
63
|
+
render(component.new(collection:, **))
|
64
|
+
end
|
65
|
+
|
58
66
|
# Construct a new query interface for filtering the current page.
|
59
67
|
#
|
60
68
|
# @param collection [Katalyst::Tables::Collection::Core] the collection to render
|
@@ -85,6 +93,11 @@ module Katalyst
|
|
85
93
|
component.respond_to?(:constantize) ? component.constantize : component
|
86
94
|
end
|
87
95
|
|
96
|
+
def default_table_pagination_component_class
|
97
|
+
component = controller.try(:default_table_pagination_component) || PagyNavComponent
|
98
|
+
component.respond_to?(:constantize) ? component.constantize : component
|
99
|
+
end
|
100
|
+
|
88
101
|
def default_table_query_component_class
|
89
102
|
component = controller.try(:default_table_query_component) || QueryComponent
|
90
103
|
component.respond_to?(:constantize) ? component.constantize : component
|
@@ -37,12 +37,12 @@ module Katalyst
|
|
37
37
|
def paginate_options
|
38
38
|
opts = @paginate.is_a?(Hash) ? @paginate : {}
|
39
39
|
opts = opts.dup
|
40
|
-
opts[:anchor_string] ||= anchor_string
|
41
|
-
opts
|
42
|
-
end
|
43
40
|
|
44
|
-
|
45
|
-
|
41
|
+
if PagyNavComponent.pagy_legacy?
|
42
|
+
opts[:anchor_string] ||= "data-turbo-action=\"replace\""
|
43
|
+
end
|
44
|
+
|
45
|
+
opts
|
46
46
|
end
|
47
47
|
|
48
48
|
class Paginate # :nodoc:
|
@@ -57,7 +57,7 @@ module Katalyst
|
|
57
57
|
def call(collection)
|
58
58
|
@collection = @app.call(collection)
|
59
59
|
if collection.paginate?
|
60
|
-
@collection.pagination, @collection.items = pagy(@collection.items, collection.paginate_options
|
60
|
+
@collection.pagination, @collection.items = pagy(@collection.items, **collection.paginate_options)
|
61
61
|
end
|
62
62
|
@collection
|
63
63
|
end
|
@@ -34,23 +34,6 @@ module Katalyst
|
|
34
34
|
def cast_value(value)
|
35
35
|
value.to_s
|
36
36
|
end
|
37
|
-
|
38
|
-
def describe_key(model, attribute, key)
|
39
|
-
label = model.human_attribute_name(attribute.name).downcase
|
40
|
-
value = model.human_attribute_name("#{attribute.name}.#{key}").downcase
|
41
|
-
|
42
|
-
description = "#{model.model_name.human} #{label} is #{value}"
|
43
|
-
description += " (default)" if default?(attribute, key)
|
44
|
-
description
|
45
|
-
end
|
46
|
-
|
47
|
-
def default?(attribute, value)
|
48
|
-
if multiple?
|
49
|
-
attribute.original_value&.intersection(cast(value))&.any?
|
50
|
-
else
|
51
|
-
attribute.default_value.eql?(cast(value))
|
52
|
-
end
|
53
|
-
end
|
54
37
|
end
|
55
38
|
end
|
56
39
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: katalyst-tables
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 3.5.
|
4
|
+
version: 3.5.2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Katalyst Interactive
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2024-07-
|
11
|
+
date: 2024-07-25 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: katalyst-html-attributes
|